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(57) Abstract 



A host adapter contains a RISC processor (210), a local memory (280), and a memory management unit (230) that permits RISC 
processor (210) and a host computer system to access local memory (280). The host computer system writes command descriptions directly 
into local memory (280). RISC processor (210) retrieves and processes the command descriptions. Local RAM (280) may be divided into 
numbered command description blocks having a fixed size and format. In standard bus protocols, such as SCSI-2, block numbers are used 
as tag messages. Such tag messages allow the host adapter to quickly identify information used when an SCSI I/O request is resumed. The 
command description blocks may be linked into lists, including an active list containing command description blocks that are ready for 
RISC processor (210) and a tree list containing command description blocks that are available for use by the host computer. 
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INTEGRATED MULT I -THREADED HOST ADAPTER 



5 BACKGROUND OF THE INVENTION 

Field of the Invention 

This invention relates to communications between a 
host computer and attached devices, and in particular 
relates to an host adapter which employs an embedded RISC 
10 (Reduced Instruction Set Computing) processor and a 

partitioned local memory to* provide an interface between a 
computer coupled to a first bus, such as a VESA bus, and 
peripheral devices coupled to a second bus, such as an 
SCSI (Small Computer System Interface) bus or an ISA bus. 

15 Description of Related Art 

Standard buses, such as ISA, EISA, VESA, PCI, and 
SCSI buses, are commonly used to create interfaces between 
the mother board of a computer and add-on devices. Often 
adapters are required between a first type of bus and a 
20 second type of bus. Fig* 1 shows a system with mother 
board 110 of a host computer 100 that communicates with 
devices 121-123 through local bus 120. Each device 121-123 
occupies a portion of the address space of host computer 



WO 95/06286 



PCT/US94/09415 



- 2 - 

100 and is identified by a base I/O port address. 

The mother board 110 contains an adapter 115 (or 
interface circuitry) for operating local bus 120. Adapter 
115 implements the protocols of bus 120 and generates 
5 signals which direct communications to the correct target 
device 121-123. 

Device 123 is an adapter between local bus 120 and 
SCSI bus 130. Peripherals 131-133 on SCSI bus 130 are 
daisy chained together and are identified by device IDs 

10 within the range from 0 to 7 or 15 if a SCSI-2 bus is 

used. SCSI controller 150 issues SCSI I/O requests to the 
attached devices 131-133 according to device ID. 

Typically, host computer 100 communicates with 
devices 121-123 and 131-133 by sending commands and I/O 

15 requests, such as a requests for a block of data from a 
hard disk, through the appropriate adapters 115 or 150. 
Most adapters require supervision by the mother board 110, 
although some functions can be completed by adapter 115 or 
150 without supervision. It is desirable to provide 

20 adapters 115 and 150 that need minimal supervision, so 
that host computer 100 can perform other operations while 
adapters 115 and 150 process I/O requests. 

SCSI controllers illustrate prior art host adapters. 
In one prior art SCSI controller, mother board 110 of host 

25 computer 100 sends an I/O request to SCSI controller 150 
by writing to a set of registers in controller 150. SCSI 
controller 150 may have several sets of registers. Each 
set of registers typically contains the number of bytes 
that can be addressed by the mother board 110. For 

30 example, if local bus 120 is a VESA bus, each device (or 
card) 121-123 attached to bus 120 occupies 16 bytes of the 
host computer's address space, and SCSI controller 150 
would have one or more 16-byte register sets. The number 
of simultaneous I/O requests that an SCSI controller can 

35 handle is typically limited by the number of register 
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sets. 

A problem with using registers to hold the I/O 
requests is that the expense of registers permits only a 
few register sets per a controller. In the register 
5 implementation, if a host computer has tens or hundreds of 
simultaneous I/O requests, the mother board must wait 
until a preceding SCSI I/O request is completed before 
sending a new I/O request. Further, a single register set 
may be too small to contain a description of a complicated 

10 I/O request. For complicated I/O requests, typically, 
further information must be requested from the host 
computer, which interrupts host computer operations and 
slows operations of the host computer, the adapter, and 
any devices attached to the host computer. 

15 In another prior art SCSI system, mother board 110 

writes a description of an I/O request into main memory 
then provides a pointer to the description. SCSI adapter 
123 uses the pointer to access the command description 
when local bus 120 is available. Typically, SCSI adapter 

20 123 copies the description from main memory on mother 
board 110 into registers in SCSI controller 150. Using 
main memory permits the mother board to write as many 
command descriptions as are need (limited by the size of 
the main memory) . However, copying creates traffic on 

25 local bus 120 and slows execution of the I/O requested 
because when SCSI bus 130 is available bus 120 may not be. 

Adapter 115 that couples mother board 110 to an ISA, 
EISA, PCI, or other standard local bus 120 experiences 
similar problems. In particular, adapter 115 often 

30 monitors and controls several simultaneous commands and 
I/O requests. If host computer 100 has another I/O 
request while adapter 115 is busy or has reached its 
capacity, host computer 100 must wait. 

Host adapters are needed which economically handle 

35 hundreds of simultaneous commands and I/O requests, which 
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minimize host supervision , and which minimize copying of 
data. 

SUMMARY OF THE INVENTION 

In accordance with the present invention, circuits 
5 and methods are provided for multi-threaded communications 
between a host computer system and devices on a bus. 
According to one embodiment of the invention, a host 
adapter contains a dedicated processor and a memory 
management unit that permits the processor and the host 

10 computer system to directly access a local memory. The 
host computer system writes command descriptions into the 
local memory of the processor where the command 
descriptions are retrieved and processed by the processor. 
RAM inexpensively provides storage for hundreds of command 

15 descriptions so that the host computer will rarely be 

delayed by limited capacity in the adapter. Further, the 
command description can be sufficiently complete that the 
processor can transmit the commands to a target device and 
process the command with minimal host intervention. 

20 Typically, the local memory is divided into command 

description blocks having a predefined size and format so 
that the starting local addresses of the command 
description blocks are multiples of a fixed quantity. The 
command description block can be numbered, and the 

25 numbers, instead of longer local addresses, can be used to 
identify the command description blocks. In standard bus 
protocols, for example SCSI-2, the block numbers can be 
used as tag messages. Such tag messages allow the host 
adapter to quickly identify the block needed when an SCSI 

30 I/O request is resumed. 

The command description blocks can be linked into 
lists, such as an active list containing command 
description blocks that are ready for the processor to 
process and a free list containing command description 
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blocks that are available for use by the host computer. 
The processor can monitor the free list for command 
description blocks written by the host computer then move 
the written blocks to the active list. Completed command 
5 description blocks can be moved from the active list to 
the end of the free list and can be used to pass to the 
host computer information concerning the completed 
command. The free and active list permits commands to be 
processed and completed in random order to increase 
10 flexibility and performance. 

BRIEF DESCRIPTION OF THE DRAWINGS 

Fig . 1 shows a system in which a host computer 
communicates with peripherals attached to an SCSI bus. 
Fig. 2 A shows an host adapter according to an 
15 embodiment of the present invention which uses a processor 
and partitioned local memory to provide a multi-threaded 
interface between a host bus and a device bus. 

Fig. 2B shows an SCSI host adapter according to an 
embodiment of the present invention. 
20 Fig. 3 shows a block diagram of a portion of a local 

memory control circuit for an SCSI host adapter according 
to an embodiment of the present invention. 

Fig. 4 shows a memory map of local memory of an SCSI 
controller according to an embodiment of the present 
25 invention. 

Fig. 5 shows a block diagram of registers used by a 
processor to provide a local address pointing to a 
location in a command description block. 

Fig. 6 shows an example free list and active list 
30 used during operation of a controller according to an 
embodiment of the present invention. 

Figs. 7A, 7B, and 7C show changes in the free list 
and active list as I/O requests are added and processed. 

Figs. 8A and 8B show a diagram of the I/O lines of an 
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SCSI controller IC according to an embodiment of the 
present invention. 

Figs. 9, 10A, 10B, 11A, 11B, 12A, 12B, 13A to 13D, 
14A to 14C, 15A to 15F, 16A to 16F, 17A to 17E, and 18A to 
5 18D show block and circuit diagrams for the SCSI 
controller of Figs. 8A and 8B. 

Figs. 19A to 19H, 20A to 20F, 21A to 21D, 22A to 22F, 
23, 24 , 25A and 25B show block and circuit diagrams of 
some of the blocks shown in Figs. 9, 10A, 10B, 11A, 11B, 
10 12A, 12B, 13A.to 13D, 14A to 14C, 15A to 15F, 16A to 16F, 
17A to 17E, and 18A to 18D. 

Similar or identical items in different figures have 
the same reference numerals or characters. 

DETAILED DESCRIPTION OF THE PREFERRED EMBODIMENTS 
15 Embodiment of the present invention provide multi- 

threaded control of devices such as peripheral devices 
attached to an SCSI bus or IDE cards attached to an AT 
bus. 

Fig. 2 A shows an adapter according to an embodiment 
20 of the present invention. The adapter is typically 

employed on the mother board of a host computer or on a 
card which plugs into a slot coupled to host bus 120. The 
adapter creates an interface between host bus 120 and 
device bus 130. Typically, the host bus is a VESA, ISA, 
25 EISA, or PCI bus so that the adapter is in the address 
space of the host computer. Device bus 130 is for 
coupling to several devices, such as IDE cards or 
peripheral devices. Device bus 130 can be but is not 
limited to an ISA, EISA, or SCSI bus. 
30 In one specific embodiment, host bus 120 is a VESA 

bus and device bus 130 is an ISA btxs. VESA bus 120 
provides a fast data transfer rate between the host 
computer and the adapter. ISA bus 130 provides a slower 
data transfer rate to one or more plug-in cards (IDE 
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devices) . In another specific embodiment disclosed in 
greater detail below, host bus 120 is a VESA bus and 
device bus 130 is an SCSI bus. 

The adapter shown in Fig. 2 A includes a host bus 
5 interface 260 and a device bus interface 250. Interfaces 
1250 and 260 create and receive signals for implementing 
the necessary protocols on busses 130 and 120 
respectively. Many types of such interface circuits are 
known in the art. A FIFO block 220 is provided to buffer 

10 data transfers such direct data transfer between host bus 
120 and device bus 130. FIFO block 220 may be omitted in 
some embodiments. 

Processor 210 is shown as a RISC processor but any 
appropriate processor or controller may be employed. 

15 Processor 210 controls the bus interfaces 250 and 260 
according to a program stored in local memory 280. The 
instruction set and the circuitry of processor 210 can be 
tailored for the functions provided and in particular, can 
be tailored for control of busses 120 and 130. 

20 Local memory interface 230 permits a host computer, 

through host bus 120 and host bus interface 260, to 
directly access local memory. The host computer writes 
command descriptions into local memory 280. Processor 210 
retrieves and processes the command descriptions. Local 

25 memory 280 is typical RAM that provides space for hundreds 
of command descriptions. 

This embodiment of the invention provides severai 
advantages when compared to adapters that employ registers 
or adapters that read command descriptions from main 

30 memory. Because local RAM is relatively inexpensively, 
space for hundreds of command description can be provided, 
and the command descriptions can be as long as necessary. 
The host computer writes the description directly into 
memory 280 and does not need to wait because registers are 

35 filled with unprocessed commands. Multiple commands for 
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each device can be queued for execution. There is no need 
for the host computer to poll the adapter to check whether 
a new command can be written and no delay before the host 
computer recognizes that another command can be written. 
5 The commands can be sent by the adapter as soon as device 
bus 130 and the target device are free. There is no delay 
waiting for host bus 120 to become free so that the 
adapter can request needed information. Because memory 
280 is local, processor 210 does hot create traffic on 

10 host bus 120 to access and execute the command 

descriptions. The adapter can use local memory 280 to 
save information when a command is disconnected and 
retrieve imformation when a command is resumed, so that 
the adapter can efficiently monitor and control 

15 simultaneous commands without host intervention. 

The ability to handle multiple commands is important 
for SCSI host adapters. As shown in Fig. 1, peripherals 
131-133 on SCSI bus 130 are daisy chained together and 
identified by device IDs within the range from 0 to 7 or 

20 15 if SCSI-2 bus is used. SCSI controller 150 identifies 
SCSI I/O requests to the attached devices 131-133 by 
device ID. ANSI X3. 131-1986, which is incorporated herein 
by reference in its entirety, defines the original SCSI 
protocol, referred to herein as SCSI-1. SCSI-1 permits a 

25 single active I/O request per device for a total of seven 
active I/O requests from the host computer. In addition, 
the host computer may have several I/O requests that must 
wait until a prior I/O requests is completed. 

A newer version of the SCSI protocol, referred to 

30 herein as SCSI-2, is defined by ANSI X3. 131-1993, which is 
also incorporated by reference in its entirety. SCSI-2 
permits multiple active I/O requests for each device. 
SCSI-2 I/O requests are identified by device ID and an 8- 
bit tag message. Accordingly, the host computer can issue 

35 up to 15 x 256 = 3840 simultaneous I/O requests all of 
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which have been started on an SCSI bus. Multiple I/O 
requests provide SCSI-2 with greater versatility and 
faster response times than SCSI-1. 

Fig. 2B shows a block diagram of an SCSI host adapter 
5 according to an embodiment of the present invention. The 
host adapter includes three separate ICs, SCSI controller 
200, local memory 280, and an EEPROM 290. In other 
embodiments, all the circuitry can be combined on a single 
IC (integrated circuit) or divided into several separate 
10 ICs. 

SCSI controller 200 can be part of an adapter card, 
such as adapter card 123 in Fig. 1, which connects to a 
local bus 120 and an SCSI bus 130 or may be provided 
directly on the mother board of a host computer where the 

15 controller 200 communicates with a CPU through a local bus 
on the mother board. Local memory 280 and EEPROM 290 are 
local to SCSI controller 200 meaning that SCSI controller 
200 can access memory 280 and EEPROM 290 directly using 
local addresses without using a shared local bus 120 of a 

20 host computer. Local storage provides faster access 

without using the resources of bus 120 or a host computer. 

SCSI controller 200 contains a host bus interface 260 
which receives and transmits signals on local bus 120. 
Local bus 120 is a VESA bus but other types of bus, for 

25 example an ISA, EISA, or PCI bus, may be used. Typically, 
host bus interface 260 contains a slave mode control 
circuit 261 to communicate with a host computer that acts 
as bus master. Slave mode control circuit 261 includes 
address decode circuit 262 which interprets I/O port 

30 address provided on bus 120 to determine if data from the 
host computer is directed to controller 200. Data latch 
and control circuit 263 is used to latch data that is 
directed to controller 200. DMA control circuit 264 is 
provided so that host bus interface 260 can perform as bus 

35 master of local bus 120 during a DMA transfer to the host 
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computer. DMA control circuit 264 includes a host address 
counter 265 to contain the address in main memory, a host 
transfer counter 266 for holding a count of the number of 
bytes transferred, and host bus master mode control 
5 circuit 267 to implement the protocol necessary to act as 
master of bus 120. The specific structure of host bus 
interface 260 depends on the kind of local bus 120 and 
protocols implemented. 

FIFO block 220 provides host FIFO 221, SCSI FIFO 222, 

10 and FIFO control circuit 223 which buffer data transfers. 
FIFO block is typically used to compensate for lack of 
synchronization of buses 120 and 130 and difference in 
data handling rates of host bus interface 260 and SCSI 
interface 250. Such FIFO blocks are often used for DMA 

15 operations and are well known in the art. 

EEPROM interface 240 provides an interface to non- 
volatile memory, EEPROM 290. EEPROM interface 240 
includes an initialization state machine 241 which 
provides initialization functions, an EEPROM control 

20 circuit 242 which provides control signals for reading 
from and writing to EEPROM 290, and a configuration 
register 243 and a data shift register 244 used in an I/O 
port address selection circuit. During initialization 
EEPROM interface 240 provides configuration data such as 

25 an I/O port base address that host bus interface 260 
compares to addresses provided on bus 120. 

SCSI interface 250 creates and receives signals on 
SCSI bus 130 and implement handshaking signals defined by 
SCSI protocols. SCSI interface 250 includes a transfer 

30 handshake circuit 251 which includes synchronous handshake 
circuit 252 and asynchronous handshake circuit 253 that 
generates signals and timing for synchronous and 
asynchronous data transfers. Included in synchronous 
handshake circuit 252 are a local storage circuit 254 for 

35 containing offset and rate data for the SCSI devices and a 
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offset control circuit 255 for keeping a count of 
unacknowledged bytes sent to an SCSI device. Control 
circuits 256 and 257 control the SCSI phase for 
arbitration, selection, and reselection according to the 
5 SCSI protocol. 

Processor 210 and the host computer access local 
memory 280 through local memory interface 230, Local 
memory interface 230 includes a memory management unit 231 
for providing control signals for local memory 280 and a 
10 data multiplexer 232 and address control 233 for selecting 
whether processor 210 or the host computer has access to 
memory. 

Memory 280 is typically RAM and partitioned to 
provide space for code and variables and space for command 

15 description blocks (CDBs) which describe SCSI I/O 

requests. Partitioning can be implemented in software by 
defining addresses which divide memory 280 into sections 
or implemented in hardware using separate RAM ICs for 
different memory areas in local memory 280. 

20 Typically, a device driver program executed by the 

host computer implements the conventions necessary for 
communication between the host computer and controller 
200. During start-up, the device driver program loads 
program code for processor 210 into local memory 280. 

25 During operation, the device driver program writes I/O 
request descriptions for SCSI controller 200 into a 
command description block in local memory 280. Data is 
written to SCSI controller 200 and local memory 280 
through VESA bus 120 using I/O port addresses which 

30 correspond to SCSI controller 200. For a VESA bus, 

controller 200 occupies sixteen I/O port addresses. To 
write to local memory 280, the host computer writes a 
local address and data to one or more of the I/O port 
addresses. 

35 The local address indicates a location in local 
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memory 280 and is written into a host address register. 234 
inside local memory interface 23 0. Data from the host 
computer goes directly into local memory 280 at the local 
address indicated by host address register 234. For 
5 writing blocks of data, host address register 234 can be 
automatically incremented (or decremented) by local memory 
interface 230 after (or before) every write to local 
memory 280 so that a single local address is sufficient 
for writing a string of data to local memory 280. 

10 The host computer reads from local memory 280 by 

writing a local address to the I/O port address that 
corresponds to host address register 234 then reading from 
an I/O port address that corresponds to local memory 280. 
To make reading of data blocks faster, local memory 

15 interface 230 automatically increments (or decrements) 
host address register 234 after (or before) every read 
from local memory 280. 

Appendix I describes an assignment of I/O port 
addresses in one embodiment of the present invention. As 

20 shown in appendix I, a word size register can be at an 
even address and a byte size register at an odd address 
even though the addresses of the registers seem to 
overlap. Words at base I/O port address plus eight and 
base I/O port address plus ten are data and local address 

25 used to read or write to local memory. In the local 
address word, fourteen bits are the local address. The 
high bits may be used for other purposes such as to 
indicate whether data is written to or read from local 
memory. 

30 Processor 210 also writes to and reads from local 

memory 280. Fig. 3 illustrates how local memory interface 
230 controls access to local memory 280. Address 
multiplexer 235 selects between two address sources, the 
host address register 234 or processor 210. Select 

35 signals for multiplexer 235 are provided by memory 
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management unit 231 on the basis of a priority system. In 
one embodiment, the host computer is always given highest 
priority so that when the host computer and processor 210 
simultaneously attempt to access memory 280, memory 
5 management unit 231 provides select signals granting 
access to the host computer. 

Data input multiplexer 232 selects the input data bus 
from which data is written to local memory 280. When the 
host computer supplies the address, VESA bus 120 supplies 

10 the data. When processor 210 supplies the local address, 
data can come from registers in processor 210 or from the 
SCSI bus 130 via SCSI interface 250. Accordingly, data 
from the SCSI bus 130 can be saved into local memory 280 
without first loading the data into a register in 

15 processor 210. 

Output data from local memory 280 is also controlled 
by the supplier of the local address. When host address 
register 234 supplies the local address, data is provided 
to the host computer on VESA bus 120. When processor 210 

20 supplies the address, data is routed either to a register 
in processor 210 or to SCSI data bus 130. 

Fig. 4 shows a partitioning of local memory according 
to one embodiment of the present invention. In Fig. 4, 
high addresses, $4000-$7FFF, of local memory are dedicated 

25 to two hundred and fifty six 64-byte command description 
blocks CDB 0 -CDB25 5 . Each command description block CDB n has 
a block number n, where 0<n£255, and a starting local 
address $4000 + (n x $40) . More generally, any starting 
address and any size command description block can used in 

30 other embodiments. Low addresses, $0000-$3FFF, contain 
local variables and a program used by processor 210. If 
two separate RAMs are provided, one for CDB memory and 
another for program memory, 14-bit addresses and enable 
signals for each RAM are sufficient to access local 

35 memory. 
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The host computer writes I/O request descriptions 
into command description blocks CDB n . 64-byte command 
description blocks provide enough memory to store 
information necessary to describe most SCSI I/O requests. 
5 For complicated scatter or gather operations, two or more 
CDBs can be linked together to describe a single I/O 
request. Larger or smaller CDB could be employed , but 
when the size of the CDB is a power of two, the block 
number n can provide a portion of the starting address of 

10 a CDB. CDB starting address are easily calculated by 

arithmetically shifting the block number n to the left and 
adding a constant if necessary. 

Processor 210 is dedicated to operations of the 
controller 200 and may be custom designed with a reduced 

15 instruction set tailored for SCSI operations and 

manipulating CDBs. Processor 210 includes an execution 
state machine 211, an arithmetic logic unit 212, an 
instruction decoder circuit 213, multiplexers 214, and a 
register set 215. 

20 Fig. 5 shows three registers from register set 215, 

instruction register 510, index register 520, and CDB 
pointer register 530, used by processor 210 to determine 
an address in a CDB. CDB pointer register 530 holds a 
block number n which indicates a CDB and provides bits six 

25 through thirteen of a 14-bit local address. CDB pointer 
register 530 can be written to from SCSI interface 250, 
from local memory 280, or by the host computer. 

When SCSI controller 200 operates SCSI-2 peripherals 
on SCSI bus 130, multiple I/O commands may be sent to a 

30 single SCSI-2 peripheral device. A device ID and an 8-bit 
tag message passed between controller 200 and the SCSI-2 
device identify each command. A block number which 
identifies a command description block can be used as the 
tag message. This provides quick identification of the 

35 correct CDB when an I/O command is resumed. The tag 
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message can be directly loaded into CDB pointer register 
530 from the SCSI bus when an I/O request is resumed . 

Least significant bits zero through five of a local 
address are an offset within a CDB and are provided either 
5 by index register 520 or instruction register 510. 

Multiplexer 540 selects which of the registers 510 or 520 
provides the least significant bits. The selection 
depends on the instruction in instruction register 510. 
For some instructions, the offset is incorporated in the 

10 instructions, . and instruction register 510 provides bits 
zero to five. For other instructions, index register 520 
provides the least significant bits of the address in a 
CDB. The offset in index register 520 can be increment 
or decremented before or after a read or write to a 

15 command description block. Appendix II provides a 

description of the instruction set used in one embodiment 
of the present invention. 

Each CDB contains fields for information which 
describes an I/O request and fields used by processor 210 

20 while an I/O request is active. Some of the fields in 
each CDB may contain include: 

1) Forward and backward pointers that link the CDBs 
into linked lists; 

2) An SCSI device ID indicating a target SCSI 
25 peripheral device to which the request is directed; 

3) SCSI command and length bytes indicating the 
operation and the number of bytes in a requested I/O; 

4) A main memory address and length which indicate 
where data transfer is directed; 

30 5) A pointer to an additional CDB for a scatter- 

gather address list used when data transfer is directed at 
several locations in main memory; 

6) A main memory address for sense data if check 
status is returned; 

35 7) Completion status bytes for indicating how much of 
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the requested I/O is complete; 

8) Status byte for indicating the status, EMPTY, 
READY, SG_LIST, ACTIVE, DISCONNECT, or DONE, of the CDB; 
and 

5 9) Storage area used during a disconnect for data 

needed when an I/O request is resumed* 

Processor 210 and the host computer keep track of 
which CDBs contain descriptions of I/O requests and which 
CDBs are available for new command descriptions. A 

10 specific method of monitoring CDBs is described below. 
Many other systems are possible and within the scope of 
the present invention. 

CDBs may be organized into a free list of CDBs 
available for new command descriptions and an active list 

15 of CDBs containing descriptions being processed by 

processor 210. Initially, all of the CDBs in local memory 
280 are in the free list and have a status byte set to 
EMPTY, a forward pointer which points to the next CDB in 
order of CDB number, and a backward pointer which points 

20 to the previous CDB. CDB^ points forward to CDB^ and CDB 0 
points backward to CDB 0 indicating the ends of the lists. 
Driver software in the host computer initializes a 
variable f irst_empty_CDB to zero indicating the first CDB 
to which the host computer can write and a variable 

25 last_empty_CDB to 255. 

When the host computer has an I/O request to send on 
an SCSI bus, the device driver . writes to the command 
description block indicated by f irst_empty_CDB , changes 
the status byte of the CDB to READY, then changes 

30 first_empty_CDB to the next CDB in the list. Processor 
210 periodically checks the free list for CDBs with status 
READY and moves the ready CDBs to the active list. The 
active list can be for example a circular linked list. 
After an I/O request described by a CDB in the active list 

35 is completed, the CDB can be removed from the active list 
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and inserted at the end of the free list. An interrupt to 
the host computer is generated so that the host computer 
checks the CDB at the end of the free list and reads 
status information of the completed I/O request. The host 
5 computer then changes the status byte of the CDB to empty 
and changes last_empty_CDB. 

After controller 200 handles several I/O requests, 
the order of the CDBs can be mixed so that forward and 
backward pointers need not point to an adjacent CDBs. 

10 Fig. 6 shows an example of a free list and an active list 
containing ten command description blocks CDB 0 -CDB 9 . The 
CDBs have addresses in memory ordered according to the 
block number 0-9. The status of each CDB (CDB 0 -CDB 9 ) is 
indicated as READY, EMPTY, DONE, ACTIVE, or SG_LIST. The 

15 logical order of the CDBs in the free list and active list 
is indicated by arrows which point from one CDB to the 
next CDB in the respective lists. For example, in Fig. 6, 
CDBj is one forward of CDB 5 in the free list, even though 
the CDBs are widely separated in address. 

20 Processor 210 uses local variables f irst_f ree_CDB and 

last_free_CDB which have initial values 0 and 255 
respectively to track of the ends of the free list. The 
f irst_free_CDB and last_free_CDB variables are closely 
related to but not always equal to the f irst_empty_CDB and 

25 last_empty_CDB variables kept by a device driver in main 
memory. The active list contains CDBs being processed by 
processor 210. At most one CDB in the active list can 
have status ACTIVE. Status ACTIVE indicates the command 
described in the CDB is currently using SCSI bus 130. All 

30 other CDBs in the active list are READY indicating an I/O 
request identified by processor 210 but not yet initiated 
on SCSI bus 130, DISCONNECT indicating an I/O request was 
initiated but the target device disconnected before 
completing the I/O request, or SG_LIST indicating a CDB 

35 containing information to be used during scatter-gather 
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functions of an ACTIVE, READY, or DISCONNECT CDB. As 
shown in Fig. 6, SG_LIST command description blocks CDB 4 
and CDB 6 are not part of the circular structure of the 
active list, but rather are pointed to by a scatter-gather 
5 pointer in CDB 9 . 

The free list contains CDBs that processor 210 has 
not yet identified as requiring any action. These include 
EMPTY CDBs that contain no command description, READY and 
SG LIST CDBs written by the host computer but not yet 

10 identified by processor 210, and DONE CDBs that processor 
210 placed at the end of the free list after completion of 
a requested I/O. 

Figs. 7A, 7B, and 7C provide examples of how the free 
list and active list shown in Fig. 6 change as I/O 

15 requests are processed. When the host computer has a new 
I/O request, the device driver writes an I/O request 
description to the command description block pointed to by 
first_empty_CDB, CDB7 in Fig. 6. If the I/O request has 
long list of addresses and transfer amounts for a scatter- 

20 gather operation, the host computer writes a scatter- 
gather list in the following command description block, 
CDB 2 , and sets a scatter gather pointer in CDB 7 to point to 
CDB 2 . As many additional CDBs as necessary may be used for 
a scatter gather list. Once the I/O request description 

25 is finished, the host computer changes the status byte of 
the CDB 7 to READY, changes the status byte of the CDB 2 to 
SGJLIST, and changes the first empty CDB variable to point 
to a CDB one forward, CDB 5 as shown in Fig. 7A. 

The host computer may write further I/O requests, for 

30 example in CDB 5 and CDB lf until f irst_empty_CDB equals 
last_empty_CDB . Since 256 CDBs are provided in the 
embodiment of Fig. 2B, this should rarely happen, but more 
that 256 CDBs can be provided if necessary to avoid delays 
while a host computers waits for an empty CDB. 
35 Processor 210 monitors the status bytes of CDBs in 
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the free list starting with f irst_f ree_CDB, CDB?. When 
processor 210 finds that the status of CDB7 is READY, the 
controller moves f irst_free_CDB forward and moves the 
READY command description block CDB? into the active list 
5 as shown in Fig. 7B. CDB7 is inserted into the active list 
by changing the forward pointer of CDB7 to point to the 
ACTIVE command description block CDB 0 and the backward 
pointer of CDB 7 to point to CDB 3 . The backward pointer of 
CDB 0 and the forward pointer of CDB 3 are changed to point 

10 to CDB7. The SG_LIST command description block CDB 2 is 
removed for the free list and is already pointed to by a 
scatter-gather pointer in command description block CDB7. 

CDBs in the active list, CDB 0 , CDB 9 , CDB 3/ and CDB7 in 
Fig. 7B, are processed by processor 210 and SCSI interface 

15 250. When the ACTIVE CDB is complete or disconnected, 
SCSI bus 130 becomes free. If no device on SCSI bus 
attempts reselection of a disconnected I/O request, 
processor 210 searches the active list for a ready CDB to 
initiate on the SCSI bus. Processor 210 can check the 

20 capabilities of a device targeted by a CDB. In 

particular, processor 210 can check to see if the target 
device is SCSI-2 compatible. If not, a CDB may be delayed 
until a previous CDB for the same device is completed. 
For SCSI-2 peripherals, processor 210 initiates an I/O 

25 request on SCSI bus 130 and provides the block number as a 
tag message. 

After an SCSI I/O request is initiated, the target 
device often disconnects while processing the request. 
This frees SCSI bus 130 for other uses. Processor 210 

30 saves information needed to resume the I/O requested in 
the disconnected CDBi then changes the status of the CDB to 
DISCONNECT. For example, processor 210 may save a main 
memory address and a remaining transfer count for an I/O 
request in the CDB describing the disconnected I/O 

35 request. 
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When a peripheral is ready to reselect an I/O request 
and SCSI bus 130 is free, the peripheral initiates SCSI 
handshaking which is responded to by SCSI interface 250. 
SCSI-2 peripheral devices return a device number and a tag 
5 message. The tag message is the block number of the 
resumed CDB. Processor 210 can quickly identify the 
address of the CDB from the tag message. With 256 CDBs , 
the CDBs are in one to one correspondence with the 
possible tag messages. SCSI-1 devices provide a device ID 

10 but do not provide a tag message. Processor 210 searches 
the active list of CDBs for the one disconnected CDB with 
the device ID. 

When a requested I/O is completed, processor 210 sets 
the status of the completed CDB to DONE, inserts the CDB 

15 at the end of the free list, and changes last_free_CDB to 
point to the inserted CDB. For example, if the ACTIVE 
command description block, CDB 0 in Fig. 7B, is completed, 
CDB 0 is moved to the end of the free list and the active 
list is reconnect into a loop as shown in Fig. 7C. Moving 

20 a CDB to the end of the free list can require the changing 
forward or backward pointers in up to four CDBs, the CDB 
moved, the last CDB in the free list, and the two CDBs in 
active list which are one forward or backward of the moved 
CDB. 

25 Processor 210 generates an interrupt for the host 

computer requesting that the host computer check completed 
CDB's. If two CDBs are completed within a short time, a 
single interrupt can request that the host computer check 
all the completed CDBs. The host computer checks the 

30 completion status of the DONE CDBs and SG_LIST CDBs 

forward of the last_empty_CDB, changes the status byte of 
the CDBs to EMPTY, clears scatter-gather pointers, then 
updates last_empty_CDB . 

Handling of the CDBs and SCSI interface 250 is the 

35 primary function of processor 210. Accordingly, the 
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instruction set of processor 210 can be tailored for these 
tasks and the circuity of processor 210 can be tailored to 
implement the instruction set. Appendix II discloses an 
instruction set for one embodiment of processor 210 for 
5 use in an SCSI host adapter in accordance with the present 
invention. A program, in the language of Appendix II, 
which implements the above disclosed handling of CDBs and 
SCSI interface 250 is disclosed in Appendix III. 

specific Embodiment of an SCSI Controller 
10 Figs. 8A and 8B shows I/O pins of an SCSI controller 

chip SEAL_1 according to an embodiment of the present 
invention. Controller chip SEAL_1 has a 24-bit address 
bus ADR and a 32-bit data bus DAT for connection to a VESA 
bus of a host computer. A 4-bit byte enable bus BE 
15 selects the bytes on data bus DAT which are used by 

controller SEAL_1. Standard VESA bus control signals as 
define in the VESA specification are handled on lines 
LADSN (local bus address strobe), LB16N (local bus size 
16-bit), LCLK (local CPU clock), LGNTN (local bus grant), 
20 BLSTN (burst transfer last) , BRDYN (burst transfer ready) , 
LREQN (local bus request), HINT (host interrupt), LDEVN 
(local bus device acknowledge), LRDYN (local bus device 
ready) , RDYRN (ready return) , ADSN (address data strobe) , 
WRN (read or write status) , MION (memory or I/O status) , 
25 DCN (data or code status) , and RTSN (system reset) . Line 
ATOSL carries a signal that enables or disable automatic 
I/O port address selection. 

I/O pins for connections to an external local memory 
(RAM or EEPROM) are provided by a 16 -bit local data bus MD 
30 and a 14-bit local address bus MA. Lines EECS, CE0N, and 
CE1N are used select whether an external EEPROM chip, a 
first RAM chip, or a second RAM chip are accessed through 
data bus MD and address bus MA. Lines CK50M and MWRN 
carry a clock signal and a read-write signal for local 
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memory . 

SCSI interface is provided through an 8 -bit SCSI data 
bus SCD and SCSI handshake lines ATNB (attention) , BSYB 
(busy), ACKB (acknowledge), RSTB (reset), MSGB (message), 
5 SELB (selection) , CDB (command or data) , REQB (request) , 
and IOB (I/O). Line SCDP controls parity chebks of the 
SCSI protocol. Such signals are well known in the art and 
described by ANSI X3. 13 1-1993 and ANSI X3. 13 1-1986. 

Lines BIOSN, ROMEN, and RAMEN control whether a basic 
10 input output system (BIOS) for the controller chip is 

loaded from local memory and whether a RAM or ROM bios is 
used. Such BIOS are well known and described for example 
in the IBM PC/AT Technic al Reference Manual published by 
IBM in 1983. 

15 Figs. 9, 10A, 10B, 11A, 11B, 12A, 12B, 13A to 13D, 

14A to 14C, 15A to 15F, 16A to 16F, 17A to 17E, and 18A to 
18D show block and circuit diagrams of controller chip 
SEAL_1. Figs. 9, 10A, 10B, 11A, 11B, 12A, 12B, and 13A to 
13D show I/O buffers for the I/O pins disclosed in regard 

20 to Figs. 8A and 8B. In Figs. 9, 10A, 10B, 11A, 11B, 12A, 
12B, and 13A to 13D, buffers IBT and IBS are input 
buffers. Buffers IBTP1 are input buffers with pull-ups to 
stop the input from floating. Buffers UOl, U02, U03, and 
U04 are output buffers. Buffer UB4 is bidirectional. 

25 Buffers UT2P2 and UT3P2 are input-output buffers with a 
pull-up on the input. Drivers DV1 and DV2 are predrivers 
for output signals. 

Fig, 10A also includes a 16-bit to 32-bit multiplexer 
1510 and a 32-bit to 16-bit multiplexer 1520 which 

30 selectably connect data bus DAT to internal data buses 
SYSDI, SYSDIL, SYSDO, SYSDOL, and SYSDOLA. In Figs. 13A 
to 13D, blocks DO_DI are historesis buffers, and parity 
generator PRTY_OUT generates a signal indicating the 
parity of SCSI output data. 

35 Figs. 14A to 14C show blocks representing a host bus 
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interface BlU.and a RISC processor RISC with accompanying 
logic and lines for signals internal to the controller 
chip SEAL_1. Block A139 is a standard 2-to-4 decoder with 
identification number A139 from "SLA1000 Series Gate Array 
5 Family Cell Library" available from S-MOS Systems, Inc. 
(the S-MOS library) . Block 910 is a 32-bit enable which 
enables or disable signals to internal data bus SYSDI. 

Host bus interface BIU implements the protocols 
necessary for communications on a VESA bus and connects to 

10 a VESA bus through the buffers shown in Figs. 9, 10A, 10B, 
11A, and 11B. Such bus interface circuits are well known 
in the art and provided on a number of commercially 
available devices which the attach to VESA buses. 

Processor RISC is tailored for control of an SCSI bus 

15 and for using the local memory and command description 
blocks as describe above* A more detailed block diagram 
of processor RISC is shown in Figs. 19A to 19H. The 
primary blocks making up processor RISC are instruction 
decoding block DECODE , a state machine block RISC_ST, and 

20 processor register block RISC_REG. Complete description 
of the blocks DECODE, RISC_ST, and RISC_REG are provided 
in Appendix IV as VHDL programs. 

Figs. 15A to 15F show circuit blocks E2P_CTL, 
CTL_REG, REG_DEC , LM_CTL, and T244. T244 is an 8-bit 

25 register from the S-MOS library. Block E2P_CTL controls 
an interface to external EEPROM including a circuit for 
selecting an I/O port address. 

Blocks CTL_REG and REGJDEC are control registers and 
register decoders. Block REG_DEC implements the I/O port 

30 addresses as described in appendix I. A complete 

description of block REG_DEC is provided as a VHDL program 
in appendix IV. A schematic of block CTL_REG is shown in 
Figs. 20A to 20F with a gate level schematic of the timer 
block TIMER from Fig. 2 OA is shown in Figs. 21A to 2 ID. 

35 Local memory control LM_CTL in Figs. 15A to 15F 
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provides and interface to local RAM attached to the I/O 
buses MA and MD. LM_CTL access local RAM through data 
buses MDO and MDI and address bus MEMADR through the 
buffer circuitry of Figs. 12A and 12B. Processor RISC 
5 from Figs. 14A to 14C accesses local RAM by providing an 
address on bus R_LM_ADR and writing data on bus RJMDI or 
reading data from bus MEM_OUT. A host computer can also 
accesses the local RAM through local memory control 
LM_CTL. Signals indicating a local address or data are 
10 provided by the host computer on I/O bus DAT and to local 
memory control LM_CTL though the buffer circuitry of Figs. 
10A and 10B via bus SYSDOL. A local address is stored in 
a register internal to local memory control LM_CTL. Data 
is written through LM_CTL to local memory via bus MDI. 
15 Data is read by the host computer via bus SYSDIL and the 
buffer circuitry of Figs. 10A and 10B. A complete 
description of block LM_CTL is provided in Appendix IV as 
a VHDL program. 

Figs. 16A to 16F and 17A to 17E show elements of an 
20 SCSI interface. SCSI interfaces are well known in the 
art and commercially available in products such as the 
AIC-7780 from Adaptec, Inc. and the NRC 53C820 which are 
both SCSI controller chips. In Figs. 16A to 16F and 17A 
to 17E, blocks T244, BLT8, T373T, and T240 are a buffer, a 
25 bus latch, a latch, and a tri-state buffer from the S-MOS 
library. Blocks SC_PRTY_IN, SCSIBLK, and SC_CTL perform 
parity checks, produce and receive SCSI handshake signals, 
and control SCSI phase. A gate level schematic of block 
SC_PRTY_IN of Fig. 16A is shown in Fig. 24. A schematic 
30 of block SC_CTL of Figs. 17 C and 17 E is shown in Figs. 25A 
and 25B. 

Figs. 22 A to 22F and 23 show a schematic of block 
SCSIBLK of Figs. 16D and 16F. Block ENC3T9 is a selector 
which selects either MDI [2:0] or SYSDI[10:8] to supply a 
35 device ID to block ARBPRO. Block ARBPRO checks priority 
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of the SCSI controller and other SCSI devices during the 
SCSI arbitration phase. In particular, block ARBPRO 
compares signals on bus SCDAT, the SCSI data bus, to 
signals on bus OWN ID to determine which device wins the 
5 arbitration. If the SCSI controller has higher priority, 
a signal on line ARBWINN indicates the controller won the 
arbitration. During selection phase, block ARBPRO checks 
if the number of bits set on the SCSI data bus is valid, 
two and only two. A device ID register in block ARBPRO 
10 indicates with which SCSI device the controller will 
comunicate. A signal on line WRDEVID writes a device ID 
from bus DIDI into the device ID register. If SELTEDB 
pulses, a device ID from bus SCDAT is written to device ID 
register. 

15 Block SELARB controls sequencing of arbitration and 

selection phases and detects SCSI bus free phase. The bus 
free phase is indicated by a signal on line BUSFREE. 
Arbiration is begun by a signal on line ENABSELB. The 
well known states in SCSI specification are implemented 

20 according to clock signals. 

Block HDSHK in Fig. 23 provides both asynchronous and 
synchronous SCSI handshake signals. A signal on line 
ENHDSHK begin SCSI Handshake protocols for both 
synchronous and asynchronous transfer. A signal on line 

25 ENSYNC differentiates synchronized or asynchronized 
handshake. For synchronous transfers, signals on bus 
RATE[2:0] determines the synchronous transfer speed. Line 
OFSSTPB carries a signal that stops synchronous transfer 
if the offset counter status does not allow further 

30 synchronous data transfer. 

For asynchronous, input SCSI request or acknowledge 
signals are provided on line REQACKI. Output SCSI 
acknowledge or request signals are provided on line 
REQACKO. Signals on line XFERCYC provide to the FIFO 

35 signals indicating data transfer. RQAKI is a one clock 
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period pulse after detection of a signal on REQACKI used 
for internal logic. 

Block OFSRATE in Fig. 23 is a local storage circuit 
that provides SCSI device offset and synchronous transfer 
5 rate information. 

Figs. 18A to 18D show blocks CNTR_DEC, EPTRCNT, 
CNT_OUT, CNTJENJ1UX, and FF_CTL which implement an SCSI 
FIFO buffer, a host FIFO buffer, and control circuitry for 
DMA transfers. Such FIFO buffers are well known in the 
10 art, and in particular, are in the commercially available 
AIC-7780 and NRC 53C820 chips mentioned above. 

Although the present invention has been described 
with reference to particular embodiments, the description 
is only an example of the invention's application and 
15 should not be taken as a limitation. The scope of the 
present invention is defined by the following claims. 
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APPENDIX I 

Bank 0 Registers 

Base adr + O — Word, Read Only — Two bytes of ASPI ID to 
identify the chip. 
5 Base adr + 1 — Byte, Read Only — One byte of ASPI ID to 
identify chip. 

Setup program finds the chip using ASPI ID before 
configuring the chip. 

Base adr + 2 — Word, Read/Write — Configuration 

10 Bit 15-12 BIOS address 

Bit 11 SCSI parity enable 

Bit 10-8 SCSI ID to be used by this chip 

Bit 7 VESA burst mode enable 

Bit 6 not used 

15 Bit 5 Host interrupt enable 

Bit 4-2 Host IRQ channel selection (not used 
by VESA) 

Bit 1-0 Host DMA channel selection (not used 
by VESA) 

20 Base adr + 4 — Word, Read/Write — More Configuration stuff 

Bit 15-14 Local memory wait state selection 

Bit 13-12 not used 

Bit 11 8 bit local memory data width 

Bit 10-8 I/O port address (high order three 

25 bits) 

Bit 7 not used 

Bit 6 Fast SCSI ACK signal 

Bit 5-0 I/O port address (low order five bits) 

The data contained in the above two registers are 
30 initialized from the EEPROM , if available, at power up. 
Changing bits 10-8 and 5-0 of base_adr + 4 changes the 
base I/O port address. To make the change effective, the 
change must be written to the EEPROM and the power 
recycled. 

35 Base adr + 3 — Byte, Read only — Chip revision number 

Base adr + 6 — Word, Read/ Write — EEPROM Data 

Base adr + 7 — Byte, Read only — EEPROM Command and Address 

These two registers are used to change the EEPROM contents 
and set up different configurations. 



40 Base adr + 8 — Word, Read/Write — Local RAM Data 

Base adr + 10 — Word, Read/ Write — Local RAM Address 
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To access the chip local RAM, the host computer writes a 
local address to the Local RAM Address register and 
follows with repeated IOR or IOW instructions written to 
high bit of the word at Base adr + 10. These registers 
5 are used to load the RISC program and the CDBs into the 
chip local memory. They can also be used to read the RISC 
program local variables during abnormal condition 





recovery. 










Base 


adr 


+ 


9 — 


-Byte, Read only — Chip Status 


10 




Bit 


7 




DMA complete 






Bit 


6 




Host FIFO ready 






Bit 


5 




Local RAM access complete 






Bit 


4 




RISC halted 






Bit 


3 




SCSI reset in 


15 




Bit 


2 




SCSI parity error 






Bit 


1 




CDB completed abnormally 






Bit 


0 




CDB completed normally 




Base 


adr 


+ 


10- 


— Byte, Write only — Interrupt Acknowledge 






Bit 


7- 


-3 


not used 


20 




Bit 


2 




Disable EEPROM auto-configuration 






Bit 


1 




Acknowledge abnormal CDB complete 












interrupt 






Bit 


0 




Acknowledge normal CDB complete 



interrupt 



25 These two registers, one read-only and one write-only, are 
typical status and interrupt registers. 

Base adr + 11 — Byte, Read/Write — Offset Register 

Base adr + 12 — Word, Read/Write — RISC Processor Program 
Counter 

30 Base adr + 15 — Byte, Read/Write — Chip Control 



Bit 7 Chip reset 

Bit 6 SCSI reset 

Bit 5 RISC halt 

Bit 4 Single step (Write) , Diagnostic 

35 failure (Read) 

Bit 3 DMA enable 

Bit 2 Timer clock select (should 0) 

Bit 1 Register bank number (0 or 1) 

Bit 0 Diagnostic bit 



40 To start the RISC program execution, both bits 5 and 4 
must be reset. To single step the RISC program, reset bit 
5 and bit 4. Bit 4 is reset by the hardware after 
executing one RISC instruction. Bit 1 is used to select 
either bank 0 or bank 1 of registers. 
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Bank 1 Registers 

This Bank 1 is not used during normal operations but may 
be used to debug the chip or a RISC program. 

Base adr + O — Word, Read/Write — RISC accumulator . 

5 Base adr + 1 — Byte, Read/ Write — RISC index register. 

Base adr + 2 — Word, Read/Write — RISC instruction register. 

Base adr + 4 — Word, Read/Write — FIFO 1,0. 

Base adr + 6 — Word, Read/Write — FIFO 3,2. 

Base adr + 8 — Word, Read/Write — DMA Address 1,0. 
10 Base adr + 10 — Word, Read/Write — DMA Address 3,2. 
Base adr + 12 — Word, Read/Write — DMA count 1,0. 
Base adr + 14 — Word, Read/Write — DMA count 3,2. 
Base adr + 3 — Byte, Read/ Write — CDB pointer. 

This register points to one of the 256 possible active 
15 CDBs. 

Base adr + 5 — Byte, Read/Write — SCSI Device ID. 

This register identifies the SCSI device the chip is 
connecting to or trying to select 

Base adr +■ 7 — Byte, Read/Write — hardware control flag. 
20 Base adr + 9 — Byte, Read/Write — SCSI Control. 



Bit 7 CD 

Bit 6 10 

Bit S MSG 

Bit 4 ATN 

25 Bit 3 Busy 

Bit 2 SEL 

Bit 1 REQ 

Bit 0 ACK 



Base adr + 11 — Byte, Read/Write — SCSI Data 
30 Base adr + 15 — Byte, Read/Write — Chip Control 
This is the same register as the one in bank 0. 
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Appendix IT 



Summary of RISC Instruction Set 





15 


14 


13 


12 


11 


10 


9 


8 


7 


6 


5 


4 


3 


2 


I 


0 


3rd & 4th bvte 


mov.b 


0 


0 


B 


r 


"r 


r 


0 


0 


i 


la 


la 


la 


la 


la 


la 


la 




mov.b 


"0 


0 


B 


r 


r 


r 


0' 


0 


o 


la 


la 


la 


la 


la 


la 


la 




mov.w 


0 


0 


w 


r 


r 


r 


0 


0 


i 


La 


la 


la 


la 


la 


la 


la 




mov.w 


0 


0 


w 


r 


r 


r 


0 


0 


0 


la 


la 


la 


la 


la 


la 


la 




movq.b 


0 


0 


B 


r 


r 


r 


0 


1 


i 




qa 


qa 


qa 


oa 

» 


Qa 


qa 




movq.b 


0 


0 


B 


r 


r 


r 


0 


1 


0 






Qa 
• 


Qa 


Qa 


Qa 


aa 




movq.w 


0 


0 


W 


r 


r 


r 


0 


1 


i 




qa 


qa 


Qa 


Qa 


Qa 

» 


Qa 




movq.w 


0 


0 


W 


r 


r 


r 


0 


1 


o 




Qa 


Qa 


Qa 


oa 


Qa 


aa 




movqx.b 


0 


0 


B 


r 


r 


r 


1 


0 


i 


0 
















movqx.b 


0 


0 


B 


r 


r 


r 


1 


0 


0 


0 
















movqx.w 


0 


0 


W 


r 


r 


r 


1 


0 


i 


0 
















movqx.w 


0 


0 


W 


r 


r 


r 


1 


0 


0 


0 
















movr 


0 


0 


B 


rd 


rd 


rd 


1 


1 


i 










rs 


rs 


rs 




movr 


0 


0 


B 


rs 


rs 


rs 


1 


1 


0 










rd 


rd 


rd 




movi.b 


0 


0 


1 


B 


r 


r 


1 


1 


I 


I 


I 


I 


I 


I 


I 


I 




movi.w 


0 


0 


1 


W 


r 


r 


1 


1 


















16 bit data 


jtstf 


1 


1 


1 


f 


f 


f 


t/f 


ja 


ja 


ja 


ja 


ja 


ja 


ja 


ja 


ja 




jtst 


1 


1 


0 


b 


b 


b 


t/f 


ja 


ja 


ja 


ja 


ja 


ja 


ja 


ja 


ja 




jcmpi 


I 


0 


1 




r 


r 


t/f 


0 


I 


I 


I 


I 


I 


I 


I 


I 


12 bit jump addr 


jcmpq 


1 


0 


1 




r 


r 


t/f 


1 






qa 


qa 


qa 


qa 


qa 


qa 


12 bit jump addr 


jmp 


1 


0 


0 


0 


ja 


ja 


ja 


ja 


ja 


ja 


ja 


ja 


ja 


ja 


ja 


ja 




call 


1 


0 


0 


1 


ja 


ja 


ja 


ja 


ja 


ja 


ja 


ja 


ja 


ja 


ja 


ja 




ret 


0 


1 


1 








1 


1 


0 


















rflag 


0 


1 


1 


f 


f 


f 


1 


1 


1 












T 


T 




dec 


0 


1 


0 








1 


1 


0 


















inc 


0 


I 


0 








I 


1 


1 


















xor 


0 


1 


0 








0 


0 


1 


la 


la 


la 


la 


la 


la 


la 




or 


0 


1 


0 








0 


0 


0 


la 


la 


la 


la 


la 


la 


la 




and 


0 


1 


1 








0 


0 




la 


la 


la 


la 


la 


la 


la 




xorq 


0 


1 


0 








0 


1 


1 




qa 


qa 


qa 


qa 


qa 


qa 




orq 


0 


1 


0 








0 


1 


0 




qa 


qa 


qa 


qa 


qa 


qa 




andq 


0 


1 


1 








0 


1 






qa 


qa 


qa 


qa 


qa 


qa 




waitfree 


0 


1 


0 






1 


1 


0 


0 


















sel 


0 


1 


0 






0 


1 


0 


0 












i/t 


at 




dma 


0 


1 


0 








I 


0 


1 


















stnt 


0 


1 


1 








1 


0 


0 


















halt 


0 


1 


1 






s 


1 


0 


1 


















RIOVX 


0 


0 


W/B 


r 


r 


r 


1 


0 


i/o 


1 
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Kef tO AN »r^wtntf/%mt 

la local memory address 

qa Q offset address (offset in current CDB) 

3 a jump address 

i input ■ 0 

0 output = 1 

i/t initiator /target, i = 1 

t/f true/false , t = 1 

B byte = 0 

W word = 1 

rrr register # 

bbb bit location in register (a) 

fff flag 

1 immediate data 

at scsi attention, set = 1 

rd register move destination 

rs register move source 

s 8=1, set interrupt 

T timer select 

Registers in RISC Processor 

a Accumulator 000 

°*P Pointer to current CDB 001 

P h SCSI bus phase register 001 

index register 010 
id_reg SCSI ID selection reconnect 010 

scsi_bus SCSI data 011 

P c Program Counter Oil 

X P°1 \ 100 
xp23 / Transfer Pointer (Host) 10 

bcOl ^ 110 
bc23 J Transfer Counter _ 
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The ASP, Inc. SASM ( SCSI Assembler ) User's Manual 
A. Introduction 

S/W Features: 

L two-pass assembler 

2. generates comprehensive information of instruction usage 

3. generates machine code at end of instruction, comments are retained 
line number of listing is the same as source file. 

4. generates information to support SCSI Symbolic Debugger. 

5. symbols and labels may be case-sensitive or case-insensitive 
instructions and directives are always not case sensetive 

6. includes a powerful constant experession evaluator. see section X 

7. supports ANSI C preprocessor directives 

8. instructions set has byte or word modifier that clearly shows 
it is a byte or word instruction 

S/W restrictions/convention : 

1. Source file line length is limited to 256 characters per line. 

2. generates one object file per one source file, 
does not create/link intermediate files. 

3. symbol and label name cannot exceed 32 characters in length 

4. label and instruction cannot be on the same line 

5. label and symbol shall always begin with an alphebet 

H/W features and limitaions: 

1. jump and compare instruction is combined. 

2. data section is 128 bytes long and starts from address 0. 
however, the last three words are reserved for special 
functions. ( to be explained later ) 

3. code section cannot exceed 4KB. 

4. there are 256 queues, from 0 to 255, accessible by setting 
queue pointer qp and the movq instruction . 

5. supports index move both from queue and data section. 

the index is automatically incremented by one when its a byte 
instruction and by two when its a word instruction. 
However the index move from data section is limited to 
the first 64 bytes. 

6. move word instruction to/from odd addess is illeagal. 

7. forward/backward relative jmp offset cannot exceeds 1022 bytes 

8. dma address range is 0 to OxTFFFFFF ( 128 MB, 27 bits address line ) 

9. supports single-stepping mode 

10. all registers are accessible from host 

11. you cannot access accumulator's MSB by using byte instruction, 
it is always acted on LSB of accumulator, MSB is undefined 
afterwards. 

12. call instruction cannot exceed two level deep, the last two 
words of data section are stacks, the stack automatically 
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wraparound, if calls exceed two level deep the stacks are 
overwritten. 

13. supports vectored time out trap, the trap use address 0x7A 
word as trap handler address, however the trap does not push 
program counter into stack, thus cannot return to point of 
interruption after its completion, the trap is treated 

more like a critical erorr handler. 

14. syncronous transfer period 100 - 340 ns 

15. syncronous REQ/ACK offset 0 - 15 bytes 



Input/Output filename convention 

*.sas - pass-one source file . 

*.!01 - pass-one output file, pass-two source file 

*.las - final listing file 

*.eas - executable object file ( with error(s) detected at pass two ) 
*.oas - executable object file ( no error ) 

Note: If errors occured in pass one, the assembler doesn't 
go to pass two. 

Introduction 

S ASM is a two-pass assembler, at first pass it substitutes 
all EQU symbols and generate the pass one output file with 
the extension name of n .!01\ It also calculates label address 
and put symbols and labels into its internal look up table. 

At pass one, die arguments and syntax of each instruction 
and directive is not analyzed, only the number of aguments 
are checked. 

There shall have no more symbols and labels to precess 
at pass two. If there is any error occured at pass one, 
the assembler stops and does not go on to pass two. 
However, the pass one output file is not deleted. 

At pass two the pass-one output is used as source file. 
Object file is generated as assembler proceeding each line, 
and resolving each symbol, the syntax is checked at this 
stage. 

If there are errors at pass two, the output file is renamed 
to extension "JEAS". listing file is retained. 



Preprocessor Directives 

#DEFINE 
#ELIF 
#ELSE . 
#ENDIF 
#ERROR 
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#DFDEF 

#IFNDEF 

#INCLUDE 

#LINE 

#UNDEF 



Assembaler directives 



DB 


define byte ( same as DC.B ) 




DC.B 


define constant byte 




DC.W 


define constant word 




DS3 


define storage byte 




DS.W 


define storage word 




DW 


define word ( same as DC.W ) 




EQU 


strings equivalence 




ORG 


set currnet data/code address ( change add 


ress increment sequence ) 



Predefined internal symbols and variables 
Register name list 



name 


mode 


size description 


ax 


r/w 


word accumulator word (16 bits ) 


al 


r/w 


byte accumulator LSB( 8 bits) 


qp 


r/w 


byte queue pointer ( 8 bits ) 


ix 


r/w 


byte queue index ( 6 bits, 0 to 63 ) 


sb 


r/w 


byte scsi bus ( 8 bits ) 


. daO 


r/w 


word dma transfer pointer, lower word 


dal 


r/w 


word dma transfer pointer, higher word 


dcO 


r/w 


word dma transfer count, lower word 


del 


r/w 


word dma transfer count, higher word 


ph 


r 


byte scsi phase ( 3 bits ) 


id 


r/w 


byte scsi id 


pc 


r/w 


word program counter 



Constant expression evaluation 

The following is numerical prefixes: 
OX, Ox : hexidecimal 0 : octal 
Ob, OB : binary V: character constant 
Note: numer may be separate by comma 



The following is binary operators: 
* : multiplication / : division 
+ : addition - : subtraction 

// : remainder ** : power 



The following is uniary operators: 
I : bitwise OR & : bitwise AND 

A : bitwise XOR [ : square root 
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+ : unsigned value - : two's compliment negate 

-rone's compliment negate !:not 
@ : which bit is on ( only one bit on allowed ) 

Hie following is binary conditional evaluation operators: 

Note: return one if condition is TRUE, return zero if condition is FALSE 

=: equal to >=: greater than or equal to 

<=: less than or equal to o: not equal to 

>: greater than <: less than 

&&and II: or 

Note: you may use ( ) to change the operation precedence 

Example: 

((5+0x0A)* (0bllll,0010 - 077))/3 is 

( 0x01 10 1 0x1001 ) & ( 0x0001 1 0x1000 ) is 0x1001 

(2**8)//13 is 

(( 100>=0xl0)&&( 0bl0U01=033) ) II ( (Oxff < 0x100) is TRUE 
@0b0100,0000 is 6 

B. Instructions set 

AND 

Operation size: Byte 
Registers: al 

Description: AND specified local memory with AL, result is in AL 
See also: ANDQ 

Example: 

andb al, byte 



ANDQ 

Operation size: Byte 
Registers: al 

Description: AND specified queue data with AL, result is in AL 
See also: AND 

Example: 

andq.b al,q[0] 



CALL 
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Registers: not applicable 

Description: push next instruction address into stack, and set 
program counter to new address, the called subroutine 
shall end with RET instruction 

Note: this instruction is used with RET instruction 

See also: ret,jmp 

Example: 

call 0x0100 
call subroutine 



DEC 

Operation size: Byte 
Registers: al 

Description: decrement AL by one, put result back to AL 
See also: INC 

Example: 

dec.b al 



DMA 

Registers: not applicable 
Description: starts DMA 



HALT 

Registers: not applicable 

Parameter none or immediate value one 

Note: a parameter of immediate zero is the same as no parameter 

Description: stop RISC CPU, user may optional send interrupt to host 

Example: 

halt 

halt #INT 



INC 

Operation size: Byte 
Registers: AL 

Description: increment AL by one, put result back to AL 
See also: DEC 
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Example: 
inc.b al 



JCMPI 

Operation size: Byte 
Condition: E .NE 
Registers: AL 

Description: compare AL with specified immediate value 

branch to new address if condition met 
Example: 

jcmpLb.e AL, #0, al_is_zero 
jcmpLb-ne AL, #0xFF r al_is_not_OxfF 



JCMPQ 

Operation size: Byte 
Condition: .E .NE 
Registers: AL 

Description: compare AL with specified queue data 
branch to new address if condition met 

Example: 

jcmpq.b.e AL, q[0], al_equals_qj) 
jcmpq.bjie AL, q[ 63 ], al_not_equals_q_63 



JMP 

Execution time: 
Machine code size: 
Instruction size: 
Registers: not applicable 

Description: move the specified new address data into program counter 
excution will begin at new address 

Example: 

jmp new_addr 



JTST 

Operation size: Byte 
Condition: .BC J3S 
Clock: 
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Machine code size: 
Registers: AL 

Description: test the specified bit is clear or set, and branch to 
new address according if condition is true 

Example: 

jtsLb.bc AL, #0, al_bitOjs_clear 
jtsLb.bs AL,#l,alJ>itlJs_not_set 



JTSTF 

Operation size: Byte 
Condition: .BC 3S 
Clock: 

Machine code size: 
Registers: AL 

Description: test the specified bit of flag register is clear or set, 
and branch to new address if condition is true 

Note: S ASM define the following flag bit to be used with the instruction 

SelectDone equ 0 ; selection phase done 

DcZero equ 1 ; dma tranfer count zero 

Selected equ 2 ; selected by target 

Reselected equ 3 ;reselectedby target 

ParityError equ 4 ; dam parity error flag set 

FreeTimerSet equ 5 ; free-running timer set, one unit time elapsed 

Example: 

jtstf.b.bc #FreeTimerSet, free_timer_not_set 
jtstf.b.bc #Reselected, idle_next_tst_target_jnode 
jtstf.b.bc #Selected, idle_next_cdb 
jtstf.b.bs #SelectDone, selection_completed 
jtstf.b.bs #DcZero, setup_status_req_wait 
jtstf.b.bc #ParityError, dc_not_zero_wait_status_in 



LODQX 

Operation size: Byte, Word 
Machine code size: 

Registers for byte instruction : AL, SB 

Registers for word instruction : AX, DAO, DAI, DCO, DC1 

Description: load AL/AX from queue by using DC register as index 

DC is automatically incremented by one or two depends 

on operation size is byte or word 

See also: MOVQX 
Example: 



WO 95/06286 



PCT/US94/09415 



- 39 - 



lodqx.b al ; same as movqx.b al, q[ix] 

lodqx.b sb ; same as movqx.b sb, q[ix] 

lodqx.w ax ; same as movqx.w ax, q[ix] 

lodqx.w daO ; same as movqx.w daO, q[ix] 

lodqx.w dal ; same as movqx.w dal, q[ix] 

lodqx.w dcO;sameas movqx.w dcO, q[ix] 

lodqx.w del ; same as movqx.w del, q[ix] 



LODX 

Operation size: Byte, Word 

Machine code size: 

Registers for byte instruction : AL, SB 

Registers for word instruction : AX, DAO, DAI, DCO, DC1 

Description: load AL/AX from local memory by using DC register as index 

DC is automatically incremented by one or two depends 

on operation size is byte or word 

See also: MOVX 
Example: 

lodx.b al ; same as movx.b al, [ix] 
lodx.b sb ; same as movx.b sb, [ix] 

lodx.w ax ; same as movx.w ax, [ix] 

lodx.w daO ; same as movx.w daO, [ix] 

lodx.w dal ; same as movx.w dal, [ix] 

lodx.w dcO ; same as movx.w dcO, [ix] 

lodx-w del ; same as movx.w del, [ix] 



MOV 

Operation size: Byte, Word 

Registers for byte instruction: AL, QP, SB 

Registers for word instruction: AX, PC, DAI, DCO, DC1 

Description: move data between local memory and register 

Exception: move data to DAO is not allowed 
you may use MOVQ.W to do it 

Example: 

mov.b al, byte 
mov.b qp, byte 
mov.b sb,byte 

mov.w ax, word 
mov.w pc,addr 
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mov.w dal, word 
raov.w dcO, word 
mov.w del, word 



MOV1 

Operation size: Byte, Word 

Registers for byte instruction: AL, PH, IX, SB 

Registers for word instruction: AX, DAI, DCO, DC1 

Description: move immediate data to registers 

Exception: move immediate data to D AO is not allowed, 
you may use MOVQ.W to do it 

Example: 

movi.b al, #0 

movi.b ph,#Oxll 

movLb ix,#12 

movi.b sb, #Oxff 

movi.w ax,#OxFFEF 

movi.w dal, #0x0010 

movLw dcO, #0x0800 

movi.w dcl,#0X0A00 



MOVQ 

Operation size: Byte, Word 

Registers for byte instruction: AL, QP, DC, SB 

Registers for word instruction: AX, PC, DAO, DAI, DCO, DO, ID 

Description: move data between queue data and registers 

Exception: although ID is a byte register, you can only use word 
instruction on it 

Example: 

movq.b al,q[0] 

movq.b qp,q[0x01] 

movq.b ix, q[ ObOOlO ] ; ObOOlO equals 2 

movq.b sb, q[ 017 ] ; 017 is a octal number, equals 15 

movq.b q[ 63 ], al ; 

movq.b q[ 62 ], qp 

movq.b q[ 61 ], ix 

movq.b q[60] t sb 

movq.w ax, q[0] 
movq.w pc,q[2] 
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movq.w id,q[4] ; ID is byte register 

movq.w daO, q[6] 

movq.w dal,q[8] 

movq.w dcO, q[ 10 ] 

movq.w dcl,q[12] 

movq.w q[0],ax 

movq.w q[2],pc . 

movq.w q[4],id ; ID is byte register 

movq.w q[6], daO 

movq.w q[8], dal 

movq.w q[10],dc0 

movq.w q[12],dcl 



MOVQX 



Operation size: Byte, Word 

Registers for byte instruction: AL, SB 

Registers for word instruction: AX, DAO, DAI, DCO, DC1 

Description: move data between queue index data and registers 
the current IX is used as index location 
K is automatically incremented by one or two depends 
on operation size is byte or word 

See also: LODQX, STOQX 

Example: 

movqx.b al, q[ix] 
movqx.b sb, q[ix] 

movqx.b q[ix], al 
movqx.b q[ix], sb 

movqx.w ax, q[ix] 

movqx.w daO, q[ix] 

movqx,w dal, q[ix] 

movqx.w dcO, q[ix] 

movqx.w dcl,q[ix] 

movqx.w q[ix],ax 

movqx.w q[ix], daO 

movqx.w q[ix], dal 

movqx.w q[ix], dcO 

movqx.w q[ix], del 



MOVR 

Operation size: Byte 
Registers: AL 

Syntax: MOVR dest_reg, src_reg 
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Description: mov data from source register to destination register 
Example: 

; use AL, QP as destination, SB, ID, IX as source 

movr.b al, sb 

movr.b al,id 

movr.b al, ix 

movr.b qp,sb 

movr.b qp, id 

movr.b qp, ix 

; use SB, ID, DC as destination, AL, QP as source 

movr.b sb,al 

movr.b sb,qp 

movr.b id,al 

movr.b id, qp 

movr.b ix, al 

movr.b ix, qp 

MOVX 

Operation size: Byte, Word 
Registers for byte instruction: AL, SB 
Registers for word instruction: AX, DAO, DAI, DCO, DC1 

Local M£Mo2>J 

Description: move data between queue index data and registers 
the current DC is used as index location 
DC is automatically incremented by one or two depends 
on operation size is byte or word 

See also: LODX,STOX 



Example: 




movx.b 


al,[ix] 


movx.b 


sb, [ix] 


movxJ> 


[ix],al 


movx.b 


[ix],sb 


movx.w 


ax, [ix] 


movx.w 


daO, [ix] 


movx.w 


dal, [ix] 


movx.w 


dcO, [ix] 


movx.w 


del, [ix] 


movx.w 


[ix],ax 


movx.w 


[ix], daO 


movx.w 


[ix], dal 


movx.w 


[ix], dcO 


movx.w 


[ix],dcl 



WO 95/06286 



PCT7US94/09415 



- 43 - 



OR 

Operation size: Byte 
Registers: AL 

Syntax: OR register, local_memory_address 

Description: OR specified local memory with AL, result is in AL 
See also: ORQ 

Example: 

or.b al,byte 

ORQ 

Operation size: Byte 
Registers: AL 

Syntax: ORQ0B) register, local_memory_address 

Description: OR specified queue data with AL, result is in AL 
See also: OR 

Example: 

orq.b al,q[ 1 ] 

RET 

Registers: not applicable 

Description: mo v a word from stack to program counter 
Note: this instruction is used with CALL instruction 
See also: CALL 



RFLAG 
Registers: not applicable 
Description: reset the specifed bit on flag register 
Note: S ASM define the following values to be used with the instruction 



ACK 


equ 


0 ;acknoledge 


ATNJDFF 


equ 


1 ; negate attention 


PARITY 


equ 


2 ; parity enor 


FTM 


equ 


3 ; free-running timer start 


WTM 


equ 


4 ; watch dog timer 


SB 


equ 


5 ; turn off scsi bus busy 


ATNON 


equ 


6 ; raise attention 
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RESET_WTM equ 0 ; turn off watch dog timer 
TOJ250MS equ 1 ;. select 250 milli-second 
TO_10SEC equ 2 ; select 10 second 
TO_lHOUR equ 3 ; select 1 hour 

Example: 

; two parameters 

rflag #WTM, #RESET_WTM 
rflag #WTM, #TO_250MS 
rflag #WTM,#TO^10SEC 
rflag #WTM,#TO_lHOUR 

; one parameter 

rflag #FTM 

rflag #ACK ; 

rflag #PARTTY ; 

rflag #ATN_OFF ; message out last byte, negate attention 

rflag #SB ; 

rflag #ATN_ON ; raise attention 



SEL 

Registers: not applicable 

Syntax: SEL (Init or Trgt), #ATN 

Description: Start SCSI arbitration, selection/reselection phase 

Example: 

sel Init,#ATN 
sel Trgt, # ATN 

SINT 

Registers: not applicable 
Syntax: SINT 

Description: set host adaptor interrupt 
Example: 
sint 

STOQX 



Operation size: Byte, Word 
Machine code size: 
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Registers for byte instruction : AL, SB 

Registers for word instruction : AX, DAO, DAI, DCO, DC1 

Description: store AL/AX to queue by using DC register as index 

DC is automatically incremented by one or two depends 

on operation size is byte or word 

See also: MOVQX 
Example: 



stoqx.b al ; same as movqx.b q[ix], al 

stoqx.b sb ; same as movqx.b q[ix],sb 

stoqx.w ax ; same as movqx.w q[ix],ax 

stoqx.w daO ; same as movqx.w q[ix],da0 

stoqx.w dal ; same as movqx.w q[ix],dal 

stoqx.w dcO ; same as movqx.w q[ix], dcO 

stoqx.w del ; same as movqx.w q[ix], del 



STOX 

Operation size: Byte, Word 

Machine code size: 

Registers for byte instruction : AL, SB 

Registers for word instruction : AX, DAO, DAI, DCO, DC1 

Description: store AL/AX to local memory by using DC register as index 

DC is automatically incremented by one or two depending 

operation size is byte or word 
See also: MOVQX 

Example: 

stox.b al ; same as movx.b [ix],al 
stox.b sb ; same as movx.b [ixj,sb 

stox.w ax ; same as movx.w [ix],ax 

stox.w daO ; same as movx.w [ix], daO 

stox.w dal ; same as movx.w [ix], dal 

stox.w dcO ; same as movx.w [ix],dc0 

stox.w del ; same as movx.w [ix],dcl 



WAITFREE 

Registers: not applicable 
Syntax: WAITFRE 

Description: wait scsi bus free 



XOR 
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Operation size: byte 
Registers: AL 

Syntax: XOR.B register, address 

Description: exclusive OR specified local memory with AL, result is in AL 
See also: XORQ 

Example: 

xor.b al»byte 



XORQ 

Operation size: byte 
Machine code size: 
Registers: AL 

Syntax: XORQ.B register, q[ index ] 

Description: exclusive OR specified queue data with AL, result is in AL 
See also: XOR 

Example: 

xorq.b al, q[ 1 ] 
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APPENDIX III 



; ASPI.SAS 

; 



By Karl Chen 



1993 



Assembler definition 



; register li'st 


ax 


egu 


al 


equ 


qp 


equ 


ix 


equ 


sb 


equ 


daO 


equ 


dal 


equ * 


dcO 


equ 


del 

7 


equ 


; tm 


equ 


ph 


equ 


id 


equ 


pc 
7 


equ 




halt optional 


INT 


equ 



:0 
:0 
:1 
:2 
:3 
:4 
:5 
:6 
:7 

:8 

:9 

:10 

:11 



accumulator word 
accumulator LSB 
queue pointer 
queue index 
scsi bus 

dma transfer pointer, lower word 
dma transfer pointer, higher word 
dma transfer count, lower word 
dma transfer count, higher word 

; = 0, free running timer counter 

» 1, scsi phase 

=2, scsi id 

= 3, program counter 



send interrupt to host 



; rf lag first parameter 

ACK equ 0 

ATN_OFF equ 1 

PARITY equ 2 

FTM equ 3 

WTM equ 4 

SB equ 5 

ATN_ON equ 6 



rf lag optional second parameter 



acknoledge 
drop attention 
parity error 
free-running timer start 
watch dog timer . 
turn off scsi bus busy 
raise attention 



RESET_WTM 
TO_250MS 
TO_10SEC 
T0_1H0UR 



equ 
equ 
equ 
equ 



0 ; turn of f watch dog timer 

1 ; select 250 milli-second 

2 ; select 10 second 

3 ; select 1 hour 



selection first parameter 

? itself is initiator 
; itself is target 
; raise attention too 



Init 


equ 


I 


Trgt 


equ 


T 


ATN 


equ 


1 


. jtstf 


flag - 




SelectDone 


equ 


0 


DcZero 


equ 


1 


Selected 


equ 


2 


Reselected 


equ 


3 


ParityError 


equ 


4 


FreeTimerSet 


equ 


5 


AtnRaised 


equ 


6 



selection phase done 
dma tranfer count zero 
selected by target 
reselected by target 
dam parity error flag set 

free-running timer set, one unit time elapsed 
current time unit is one second 
attension raised 
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QS_FREE 


equ 


0x00 


7 




QS_READY 


equ 


0x01 


• 




QS_DISC 


equ 


0x02 


• 

9 




QSJ3USY 


equ 


0x04 


• 
9 




QS_ACTIVE 


equ 


0x08 


• 

§ - 




QS_DATA_XFER 


equ 


0x10 


• 

9 




QS_DONE 


equ 


0x80 


7 




t 

QC_POST • 


equ 


0x01 


7 




QC LINK 


equ 


0x02 


• 

9 




QC_SG_LIST 


equ 


0x04 


7 




QC_DATA_IN 


equ 


0x08 


9 




QC_DATA_OOT 


equ 


0x10 


• 




QC_MSG_IN 


equ 


0x20 


; wait 


message in after message out 


QC_MSG_OUT 


equ 


0x40 


; do message out after selection completed 


QC_REQ_SENSE 


equ 


0x80 


7 




; QC_DO_TAG_MSG 


equ 


0x10 


7 




; completion status 


r qt 


done_ 


stat ] 




QD_NOJ3RROR 




equ 


" 0x00 j 


; command done without error 


QD_BAD_CDB_TRG_ID 




equ 


0x01 j 


; illeagal target id 


QD SELECT TIMEOUT 




equ 


0x02 ; 


» selection phase time out 


QD_NO__CMD_XFER 




equ 


0x03 : 


; no command transfer phase 


QD_NO_DATA_XFER 




equ 


0x04 ( 


; no data transfer phase 


QD_DATA_XFER_UNDER_RUN 


equ 


0x05 


; DMA data UnderRun, xfer count not 


QD_CAN_NOT_GET SENSE 


equ 


0x06 




QD_B AD_SCS I_STATUS 




equ 


0x07 




QD_WTM_TIMEOUT 




-equ 


0x08 


? watch-dog timer time out 


7 

; queue link q[ fwd 


] & 


q[ bwd ] field definition 


QLINK_TAIL 




equ 


OxFF ; 


9 

QUEUE_SIZE 




equ 


0x40 ; 



SCSI command queue, shall be the same as ASPI Host driver 



#define fwd 0 

#define bwd 1 

#def ine cntl 2 

#define sg_entry_cnt 3 

; for sg_list only 

9 

#define sg_list0 4 

#define sg_listl 6 

#define sg_list2 8 

#define sg_list3 10 



for non-sg_list only 



#def ine cdb_len 4 

#define target_id 6 

#define target_lun 7 

#define done_stat 8 

#define scsi_stat 9 

#define scsi_msg 10 

#define reserved 11 



q forward pointer 

q backward pointer 

q control status 

number of sg entry of the queue 

meaningful for sg_list only 



first sg list entry address 
second sg list entry adress 
third sg list entry address 
fourth sg list entry address 



SCSI command length 
Target SCSI ID 

Target SCSI logical unit number 

q completion status 

SCSI command completion status 

command done message 

reserved 
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field 12-20 is for sg_list head queue only 



#define sg_list_qp 
#define sg_list_fwd_qp 
#define sg_page_si2e 
last ) 

#define sg_first_xfer_cnt 
#define sg_last_xfer_cnt 



12 j 

13 ; 
14 



the first sg_list queue pointer 
the working sg_list queue's forword pointer 
} sg entry transfer count ( except first and 



16 ; sg entry first transfer count 
IB- ; sg entry last transfer count 



field 12-20 is for non-sg_list only 



#def ine data_cntO 
#define data_cntl 
#define data_addr0 
#defxne data addrl 



#define 
#define 
#define 
#define 
#define 
#define 
#define 
#define 
#define 
#def ine 
#define 
#define 



sense_len 
sense_addrO 
sense_addrl 
cdb 

busy_loop 

i ni t_bu sy_loop 

busy_retry 

timeout_chk 

timeout_cnt0 

timeout_cntl 

msg_out_code 

tag_code 



#define status 



12 
14 
16 
18 

20 
22 
24 
26 
38 
39 
40 
41 
42 
43 
44 
45 



46 



dam transfer count low word 
dam transfer count high word 
dam transfer address low wor 
dam transfer address high wor 

SCSI request sense data length 
request sense message buffer low word 
request sense message buffer high word 
SCSI CDB block, 12 bytes maximum 

set busy_loop to this value when expired 

if zero, no disconnect timeout check 
free-running timer count LSB 
free-running timer count MSB 

first byte of queue tag messasge 
either 0x20, 0x21, 0x22 

second byte is queue tag, input active_cdb 
q current execution status 
this shall be the last byte 



host need not initialize data after here 



#define 
#define 
#define 

#define 
#define 
#define 
#define 
#define 
#define 



x_saved_sg_entry_cnt 47 
x_saved_ sg_index 
x_saved_sg_list_qp 



48 
49 



x_reconnect_rtn 50 

x_saved_data_cntO 52 

x_saved_data_cntl 54 

x_saved_data_addr 0 5 6 

x_saved_dat a_addr 1 58 

reserved2 60 



the current working sg_list q pointer 



reserved 



SCSI status bytes 
bits of status byte 



6 5 4 



3 2 



Status 



RR0 0 000R =0x00 GOOD 

RR00 001R =0x02 CHECK CONDITION 

RR00 010R =0x04 CONDITION MET 

RR00 100R =0x08 BUSY 
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R R 0 1 OOOR =0x10 INTERMEDIATE 

RR 01 010R =0x14 INTERMEDIATE -CONDITION MET 

RR01 100R =0x18 RESERVATION CONFLICT 

RR10 001R =0x22 COMMAND TERMINATED 

RR10 100R =0x28 QUEUE FULL 



All Other Codes 

bit 0, 6, 7 are reserved 



Reserved 



ss 


_GOOD 


equ 


0x00 


ss" 


~CHK_CONDITION 


equ 


0x02 


ss" 


_CONDITION MET 


equ 


0x04 


ss" 


TARGE T_B US Y 


equ 


0x08 


ss" 


_INTERMID 


equ 


0x10 


ss. 


_INTERMID_COND_MET 


equ 


0x14 


ss 


_RSERV_CONFLICT 


equ 


0x18 


ss" 


~CMD_TERMINATED 


equ 


0x22 


ss„ 


_QOEUE_FULL 


equ 


0x28 



target has successfully completed the command 
contigent allegiance condition has occur ed 
the requested operation is satisfied 
target is busy 
intermediate 

intermediate- condition met 

the combination of condition-met ( 0x04 ) 

and intermediate ( 0x10 ) statuses 

reservation conflict 

command terminated 

by terminated I/O process message or 

a contigent allegiance condition has occured 

queue full 



MSG C/D I/O 



PH_DATA__OUT 


equ 


(ObOOOO) j 


0 


0 


0 


I -> T, data out 


PH_DATA_IN 


equ 


(ObOOOl) j 


; 0 


0 


1 


T -> I, data in 


PH CMD OUT 


equ 


(ObOOlO) 


• 0 


1 


0 


I -> T, command 


PH_STAT_IN 


equ 


(ObOOll) 


; 0 


1 


1 


T -> I, status 


PH_RES1 


equ 


(ObOlOO) , 


r 1 


0 


0 


reserved ' 


PH_RES2 


equ 


(ObOlOl) 


f 1 


0 


1 


reserved 


PH_MSG_OUT 


equ 


(ObOllO) , 


f 1 


1 


0 


I -> T, message in 


PH_MSG_IN 


equ 


(ObOlll) 


; 1 


1 


1 


T -> I, message out 



scsi messages 
MS_CMD_DONE equ 
MS_EXTEND equ 



0x00 ? command completed 

0x01 ; first byte of extend message 



one byte messages, 0x02 - OxlF 
0x12 - OxlF: reserved for one-byte 



messages 













I T, I-initiator T-target support 












O: Optional, M mandatory 


M1_SAVE_DATA_PTR 


equ 


0x02 ; 


O 


o 


save data pointer 


M1_RE STORE_PTRS 


equ 


0x03 ; 


O 


0 


restore pointers 


Ml_DISCONNECT 


equ 


0x04 ; 


O 


0 


disconnect 


M1_INIT_DETECTED_ERR 


equ 


0x05 ; 


M 


M 


initiator detected error 


Ml_ABORT 


equ 


0x06 ; 


O 


M 


abort 


M1_MSG_REJECT 


equ 


0x07 j 


M 


M 


message reject 


Ml_NO_OP 


equ 


0x08 ; 


M 


M 


no operation 


M1_MSG_PARITY_ERR 


equ 


0x09 j 


M 


M 


message parity error 


Ml_LINK_CMD DONE 


equ 


OxOA ; 


» O 


O 


link command completed 


Ml_LINK_CMD_DONE_WFLAG 


equ 


OxOB 


; o 


0 


link command completed with flag 


M1_BUS_DVC_RESET 


equ 


OxOC 


f 0 


M 


bus device reset 


Ml_ABORT_TAG 


equ 


OxOD 


) o 


O 


abort tag 


Ml_CLR_QUEUE 


equ 


OxOE 


r o 


0 


clear queue 


Ml_INIT_RECOVERY 


equ 


OxOF 


? o 


0 


initiate recovery 


M1_RELEASE RECOVERY 


equ 


0x10 


? o 


0 


release recovery 
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Ml KILL 10 PROC 



equ 0x11 ; 0 O terminate i/o process 



first byte of two-byte queue tag messages, 0x20 - 0x2F 
queue tag messages, 0x20 - 0x22 
M2_QTAG_MSG_SIMPLE equ 0x20 

M2_QTAG_MSG_HEAD equ 0x21 

M2_QTAG_MSG_ORDERE D equ 0x22 

M2_IGNORE_WIDE_RE5IDUE equ 0x23 
0x24 - 0x2F: reserved 
0x30 - 0x7F: reserved 
0x80 - OxFF: identify message 



O O simple queue tag 
0 0 head of queue tag 
O O ordered queue tag 
O O ignore wide residue 



extended message, first byte is 
0x04 - 0x7F: reserved 
0x80 - OxFF: vendor unique 



0x01 



MX_MODIFYJDATA PTS 


equ 


0x00 


7 


0 0 modify data pointer 


MX_S YNC_DATA_XFER_RE Q 


equ 


0x01 


7 


O O syncronous data transfer reques 


MX_SCSI1_IDENTIFY 


equ 0x02 


; O O reserved, used for SCSI-1 


identify message 










MX_WIDE_DATA_XFER_REQ 


equ 


0x03 


• 
/ 


O O wide data transfer request 


J 

MS_MIN_lBYTE 


equ 


0x02 


t 


0x02 - OxlF 


MS_MAX_lBYTE 


equ 


OxlF 


7 


0x02 - OxlF 


MS_MIN_2BYTE 


equ 


0x20 


7 


0x20 - 0x2F 


MS_MAX 2BYTE 


equ 


0x2F 


7 


0x20 - 0x2F 


MS_MIN_IDENTIFY 


equ 


0x80 


• 

r 


identify message ( over 0x80 ) 


; identify message bit 


setting 






IM_IDENTIFY MSG 


equ 


0x80 




bit 7, identify message 


IM_DISC_PRIV 


equ 


0x40 




bit 6, allow disconnect priviledge 


IM_LUN_TAR 
7 


equ 


0x20 




bit 5, logical unit target 


MASKJLUN 


equ 


0x07 




to get only bit 0-2, LUN field 


SG_ENTRY_PER CDB 


equ 


OxOF 




MAX_B U S Y_RE TR Y 


equ 


0x10 






SELECTION TIMEOUT 


equ 


0x10 






MAXJTIME OUT 


equ 


OxFF 






NULL 


equ 


(0) 




null pointer 


ZERO 


equ 


(0) 






ONE 


equ 


( ZERO 


+ 1 ) 



extended 



ERR_D I S C JTIME OUT 

ERR_TARGET_MODEJNOT_SUPPORTED 

ERR_RECONNECT_NO_MSG_IN__l 

ERR_RECONNECT_NO_MSG_IN_2 

ERR__RE CONNE CT_B AD_I D_MSG 

ERR_S C S 1 2 _RE CONNE CT_B AD_QT AG 

ERR_SCSI2_REC0NNECT_BAD_QSTAT 

ERR_SCSI2_R£CONNECT_BAD3 

ERR_SCSI2_RECONNECT_BAD4 

ERR_SCSI2_RECONNECT_BAD_ID 

ERR_SCSI2_RECONNECTJBAD_LUN 

ERR_SCSI l_RECONNECT_BAD 

ERR_TARGET_NOT_SUPPORTED 

ERR_NO_ID_MSG_AT_SELECT 

ERR NO TAG MSG AT SELECT 



equ 


0x0001 ; 


equ 


0x0002 ; 


equ 


0x0003 ; 


equ 


0x0004 ; 


equ 


0x0005 ; 


equ 


0x0006 ; 


equ 


0x0007 ; 


equ 


0x0008 ; 


equ 


0x0009 ; 


equ 


OxOOOA ; 


equ 


OxOOOB ; 


equ 


OxOOOC ; 


equ 


OxOOOD ; 


equ 


OxOOOE ; 


equ 


OxOOOF ; 
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ERRJNO_CMD_OUT equ 0x0010 

ERR_CMD_OUT_INCOMPLETE equ 0x0011 

ERR_NO_DATA_PHASE equ 0x0012 

ERR_SENSE_XFER — INCOMPLETE equ 0x0013 

ERR_INVALID_DATA__IN_PHASE equ 0x0014 

ERR_INVALID_DATA_OUT_PHASE equ 0x0015 
ERR__ABNORMAL_END_OF_DATA_XFER equ 0x0016 

ERR_NO_STAT_IN ~~ equ 0x0017 

ERR_NO_GMD_CMPL_M5G equ 0x0018 

ERRJBUSYJRETRYJTIMEOUT equ 0x0019 

ERR_RE SE LE CT_SCS 1 2_RTN equ OxOOlA 

ERR_RESELECT_SCSI1_RTN equ OxOOlB 
; date: 5/25/93 

ERR_CMD_DONE_MSG equ 0x0 01C 

ERR_L I NK_CMD_DONE equ OxOOlD 

ERR_L INK_CMD_DONE_WFIAG equ OxOOlE 

ERR_S G_L I S T_N 0_D AT A_XFER equ OxOOlF 

ERR_P ARI T Y_D ATA_IN equ 0x0020 

ERR_M1_MSG_IN equ 0x0021 

ERR_M2_MSG_IN equ 0x0022 

ERR_UNKNOWN_MSG_IN equ 0x0023 

ERR_NO_MSG_IN_AT_EXTMSG equ 0x0024 

ERR_S G_L I ST_OVER_FLOW equ 0x0025 

ERR_SG_LI ST_UNDER_FLOW equ 0x0026 

ERR_DONE_LINK_CORRUPTED equ 0x0027 

ERR_E XT_M S G_I N_ERROR 1 equ 0x0028 

ERR_EXT_MSG_IN_ERROR2 equ 0x0029 

ERR_UNKNOWN_MSG_IN_0 1 equ 0x002A 

ERR_RAISE_ATN_FAILED_01 equ 0x002B 

ERR_RAISE_ATN_FAILED_02 equ 0x002C 

ERR_SAVE_DATA_PTR_STATUS equ 0x002D 

ERR_RES_DATA_PTR_STATUS equ 0x0 02E 

ERR_DATA_XFER_OVER_RUN equ 0x0 02F 



overrun, target still in data phase 
we have to reset target 



ERR__WTM_JT I ME OUT 

r 

HALT_EXT_MSG 

. *************************** 



equ OxOOFF ; 

equ 0x0100-; 
******************************************* 



YEAR equ 1993 ; year shall be greater than 1990 

MONTH equ 6 ; valid data 0-15 

DAY equ 14 ; valid data 0-31 

VER_MAJOR equ 1 ; major version number 

VER_MINOR equ 0 ; minor version number 

? 

CODE_BEG equ 0x80 ; 

VECT_BEG equ ( (CODE_BEG) - (2*3) ) ; 

DATA_BEG equ ( (CODE_BEG) - 0x40 ) ; 



ORG ZERO 



DATA SECTION: 



SYNJCFER equ 
MSGIN_BUF_SIZE equ 



( (QUEUE_SIZE)- CMD_REQ_LEN ) 
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syn_msgjbuffer : 

ext_msg_beg 

ext_msg_len 

ex*tjnsg_req_code 

ext_msg_xfer_period 

extjns g__o f f se t 



dc.b 


0x01 


dc.b 


0x03 


dc.b 


0x01 


dc.b 


25 


dc.b 


15 



0x01 is extended message 
message length 

0x01 is syncronous negotiation 
4 ns per unit number 



#endif 



CMD REQ0 


equ 


0x03 ; 


? request sense command 


code 


CMD REQ1 


equ 


0x00 ( 


? lun bit 7-5, reserved 


bit 4-0 


CMD_REQ2 


equ 


0x00 ; 


; reserved 




CMD_REQ3 


equ 


0x00 , 


; reserved 




CMD_REQ4 


equ 


0x00 j 


? allocation length 




CMD_REQ5 


equ 


0x00 ; 


; control 




CMD_REQ_LEN 


equ 


0x06 






r 

ORG 


( (DATA_BEG) - CMD_REQ_LEN ) 





f ~" 

cmd__req_sense 


dc.b 


CMD_REQ0 , 


; request sense command 


code 




dc.b 


CMD_REQ1 


\ lun bit 7-5, reserved 


bit 4-0 




dc.b 


CMD_REQ2 


) reserved 






dc.b 


CMD REQ3 


* reserved 






dc.b 


CMD_REQ4 


; allocation length 






dc.b 


CMD_REQ5 


? control 






# 

ORG DATA_ 


_BEG 








halt_code 


dew 


0x0000 







day: bit 0-4, 5 bits 
month: bit 5-8, 4 bits 
year: bit 15-9, 7 bits 



code_chk_sum dew 
version_date dew 
| ( (DAY)fiOxOOlF) ) 
version_no 



0x0000 ; machine code check sum 

( ((<<YEAR)-1990)«9)*0xFE00) | (( (MONTH )«5 ) &Ox01E0 ) 



dc.b ( ( ( (VER_MAJOR) & OxOF) « 4) | ((VER_MINOR) & OxOF) ) 



host_scsi_id 

risc_next_ready 

risc_done_next 

scsi2_enable 

scsil_busy 

total_cdb_cnt 

active_cdb 

reconnect_lun 

saved_active_cdb 

next_active_cdb 
sync_negotiation 



dc.b 
dc.b 
dc.b 
dc.b 
dc.b 
dc.b 
dc.b 
dc.b 
dc.b 

dc.b 
dc.b 



0x80 
0x00 
0x07 
OxFF 
0x00 
0x00 
0x00 
0x00 
OxFF 

0x00 
0x00 



should be set by host 
tail of done queue 
head of ready queue 

bit set if corresponding device is SCSI II 
bit set if corresponding device is busy 



used to check if active_cdb was no process 
because of re-connection 



; above data address should not be changed 
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tempq 
count 
dummy 



dc.b 0x00 
dc.b 0x00 
dc.b Oxaa 



ORG (VECT_BEG - 2) 
WTM_SEL_TIMEOUT equ 0x01 
7 

wtm_f lag 

ORG 
wtm vect: 



; is selection time out 



stackO 
stackl 



dew 0x0000 
VECT_BEG 

jmp wtm_isr ; watch dog timer timeout ISR 

dew 0x0000 

dew 0x0000 



; 



ORG 

CODE_SECTION: 

code_beg: 

movi.w 
mov.w 



CODE BEG 



ax, #ZERO 
wtin_flag, ax 



; watch dog timer ISR vector 



. ============ 

idle_no_cdb : 
mov.b 
movi.b 



qp , risc_next_ready 
al, #QS_READY 



idle_no_cdbl : 

jcmpq.b.ne al, q[ status ], idle_no_cdbl 



movi.b al, #QLINK_TAIL 

mov.b saved_active_cdb , al 

mov.b active_cdb, qp ; save next ready queue to active_cdb 

movi.b al, #ONE 

mov . b tot al_cdb_cnt , a 1 

movq.b al, q[ cntl ] 

jtst.b.bs al, #8QC_SG_LIST, idle_nolcdb_sg_list 

next ready queue is not sg_list 

movq.b al, q[ fwd ] 

mov.b risc_next_ready, al 

jmp idle no cdb link itself 

. — z — z — z 

; next ready queue is a sg_list 

; 

idle_no_cdb_sg_list : 
7 

} is sg_list, we set up q[ sg_list_qp ] 
7 

movq.b al, q[ fwd J ; save first sg_list in al 

movq.b q[ sg_list_qp ] , al ; first sg_list queue 

movq.b q[ x_saved_sg_list_qp J , al ; working sg_list queue 



search and mark end of sg_list, testing QC_SG__LIST bit 
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; also set risc_next_ready 
7 

idle_no_cdb_srh_sg_tail : 

movq.b qp, q[ fwd ] 

movq.b al, q[ cntl ] 

jtst.b.bs al f #8QC_SG_LIST, idle_no_cdb_srh_sg_tail 

r 

; the sg_list has QC_SG_LIST flag set, its q[fwd] is not terminated yet ( OxFF 
) ' 

; if it is terminated, we cannot find the next ready queue 

9 

mov.b risc_next_ready, qp ; set risc_next_ready to q[fwd] of last 

sg_list 

movq.b qp, q[ bwd J ; restore qp to tail of sg_list 

movi.b al, #QLINK_TAIL 

movq.b q[ fwd ] , al ; host doesn't set LINK END, we must set it 

mov.b qp, active_cdb ; restore qp to active_cdb 

i 

} set up q[ sg_list_fwd_qp ], must after sg_list being terminated by 
QLINK_TAIL 

? . 

movq.b qp, q[ fwd ] 

movq.b al, q[ fwd ] ; working sg_list queue's forward queue 

mov.b qp, active_cdb 

movq.b q[ sg_list_fwd_qp ] , al ; the forward pointer of working 

sg_list 

7 

idle_no_cdb_link_itself : 

mov.b al, active_cdb 

movq.b q[ fwd ] , al ; link pointers to itself 

movq.b q[ bwd ] , al 

jmp ready_cdb_found 



idle_next_cdb : 

movi.b al, #QLINKJTAIL 

mov.b saved_active_cdb, al 

rflag #WTM, #RESET_JWTM 

j tstf .b .be #FreeTimerSet , f ree_timer_not_set 

rflag #FTM 

call dec__timeout__cnt 

r 

free_timer_not_set: 

movi.b al, #QS_READY 

jcmpq.b.e al, q[ status ], ready_cdb_found 

r 

movi.b al, #QS_DISC 

jcmpq.b.ne al, q[ status ], idle_next_test_busy 

the queue is disconnected 

check disconnect time out 

; 

movi.b al, #ZERO 
jcmpq.b.e al, q[ timeout_chk ], test_next_ready ; no time out 
checking needed 
; 
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jcmpq.b.e al, q[ timeout_cntl ] f di9cjtimeout_cntl_2ero 
jmp test_next_ready 

} 

disc_timeout_cnt l_zero : 

jcmpq.b.e al, q[ timeout_cntl J, disc_timeout_cntl_zero 
jmp test_next_ready 

? 

disc_timeout_cntO_zero : 

movi.w " ax, #ERR_DISC_TIMEOUT ; disconnection time out 1 
jmp error_halt 

7 

idle_next_ testjbusy : 

movi.b al, #QS_BUSY 

jcmpq.b.ne al, q[ status ], test_next_ready 

the queue is busy, decrement busy_loop, 
when it reach ZERO, test device ready 

movq.b al, q[ busy_loop ] 

dec.b al 

jcmpi.b.e al, #ZERO, idle_next_test_busyl 

movq.b q[ busy_loop ], al 

jmp test_next_j:eady ; busy loop not expired yet 

; while retry not expired, keep trying 
t 

idl ejaext_te st_bu sy 1 : 

movq.b al, q[ init_busy_loop ] 

movq.b q[ busy_loop ] , al ; set q[busy_loop] to init loop 

value 

movq.b al, q[ busy_retry J 

jcmpi.b.e al, #ZERO, ready_cdb_found ; if ZERO, always retry, no 

timeout 

dec . b al 

jcmpi.b.e al, #ZERO, idle_busy_timeout ; loop expired, try again 
movq.b q[ busy_retry ] , al ; decrement q[busy_retry] 

Note: q[busy_retry] is not decrmented to zero when timeout 

idle__busy_timeout: 

movi.w ax, #ERR_BUS Y_RE TR Y_TIME OUT 

jmp error_halt 

r 

test_next_ready : 

mov.b qp, risc_next_ready 

movi.b al, #QS_READY 

jcmpq.b.ne al, q[ status ], idle_next_test_f lag ; you should restore 
active cdb 



next queue is ready 

mov . b qp , act ive_cdb 

movq.b al, q[ fwd } 

mov.b tempq, al ; save tempq = active_cdb->q_fwd 

mov.b al, risc_next_ready 

movq.b q[ fwd ] , al 



mov.b 



qp, tempq 
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mov.b al, risc_nextjready 

movq.b q[ bwd J , al 

mov.b qp, risc_next_ready 

mov . b al , active_cdb 

movq.b q[ bwd J, al 

mov.b active_cdb, qp 

the risc_next_r'eady is linked into the active queue * 
now let risc_next_ready be the active_cdb 

mov.b al, total_cdb_cnt 

inc.b al 

mov . b t o ta l_c db_cnt , a 1 

movq.b al, q[ cntl ] 

jtst.b.bs al, #§QC_SG_LIST, idle_next_sg_list 

reset risc_next_ready, no sg_list 

movq.b al, q[ fwd ] 

mov.b risc__next_ready, al 

mov.b al, tempq 

movq.b q[ fwd J, al 

jmp ready_cdb_f ound 

is sg_list, we set up q[ sg_list_qp ] & q[ x_saved_sg_list_qp 1 

idle_next_s g_li st : 

movq.b al, q[ fwd J 

movq.b q[ sg_list_qp ] , al 

movq.b q[ x_saved_sg_list__qp ], al 

search and mark end of sg_list, also set risc_next_ready 
Note: after search ends, the qp is not at end of sg_list 

idle_next_srh_sg_tail : 

movq.b qp, q[ fwd 1 

« 

r 

movq.b al, q[ cntl J 

jtst.b.bs al, #eQC_SG_LIST, idle_next_srh_sg_tail 

found next queue of sg_list end, now set risc_next_ready to the qp 

mov . b r isc_next_r eady , qp 

movq.b qp, q[ bwd 1 ; restore qp to end of sg_list 

movi.b al, #QLINK_TAIL 

movq.b q[ fwd ] , al ; mark end of SG__LIST 

mov.b qp, active_cdb 

set up q[ sg_list_fwd_qp ] 

movq.b qp, q[ fwd ] 

movq.b al, q[ fwd ] ; got working sg_lisfs forward pointer 

mov . b qp , act ive_cdb 

movq.b q[ sg__list_fwd_qp ], al 

» 

; now we link the sg_list header q[ fwd ] 
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4 



mov.b al, tempq ; tempq is last active_cdb->q_fwd 

movq.b q[ fwd ], al 

jmp ready_cdb_found 

• — — — — — — — — 

idle_next_test_f lag : 

mov.b qp, active_cdb ; restore active_cdb 

jtstf .b.bc #Reselected, idle_next_tst_target_mode 

jmp reselect_found 

idle_next_tst_tar getmode : 

jtstf.b.bc #Selected, idle_jnext_cdb 



scsi_target_mode : 

rflag #WTM, #RESET_WTM 

movi.w ax, #ERR_T ARGE T_MODE_NOT_S UPPORTE D 

jmp error_halt 



; =============== 

reselect_f ound : 
rflag 
jcmpi.b.e 
movi.w 
jmp 



reselect_id__msg : 
movr.b 
rflag 
jtst.b.bs 



#WTM, #RESET_WTM 
ph, #PH_MSG_IN, reselect_id_msg 
ax, #ERR_RECONNECT_NO_MSG_IN_l 
error halt 



; first byte of message in phase 



al, sb 
#ACK 

al, #8lM_IDENTIFY_MSG, reselect_chk_scsi2 ; IDENTIFY mesage 
bit 7 always on, 0x80 - OxFF 

movi.w ax, #ERRJR£CONNECT_BAD_ID_MSG 

jmp error_halt 



reselect_chk_scsi2 : 
J 

; next 4 instructions to accomplish anding reconnect_lun with 0x07 I 1 1 

mov.b reconnect_lun, al ; al contains sb, message in first byte 

movi.b al, #MASK_LUN 

and.b al, reconnect_lun 

mov.b reconnect_lun, al 

* 

movr.b al, id 

and.b al, scsi2_enable 

jcmpi.b.e al, #ZERO, reselect_scsil 

jcmpi.b.ne ph, #PH_MSG_IN, reselect_scsil 

t 

movr.b al, sb ; second byte of message in phase 

rflag #ACK 

jcmpi.b.e al, #M2_QTAG_MSG_SIMPLE, reselect_q_simple 
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movi.w 
jmp 
7 

reselect_(i_siinple : 
jcmpi.b.e 
movi.w 
jmp 



ax, #ERR_S CS 1 2_RECONNE CT_BAD_QTAG 
error halt 



ph, #PHJiSG_IN, reselect_chk_status 
ax, #EKR_RECONNECT_NO_MSG_IN_2 
error halt 



reselectjchk_status : 

movr.b qp, sb ; third byte, must set qp to verify new qp's 

q[ status] 

rflag #ACK 

movi.b al, #QS_DISC 

jcmpq.b.e al, q[ status J, reselect2_chk_id ; check if q{status] 
contain valid status 



movi.w 
jmp 
7 

reselect2_chk_id : 
movr.b 
j cmpq • b . e 
movi .w 
jmp 

7 

reselect2_chk_lun : 
mov.b 
j cmpq • b . e 
movi.w 
jmp 

7 

re9elect2_f ound : 
mov.b 
mov.b 
mov.b 
movq .w 



ax, #ERR_S C S 1 2 __RE CONNE CT_B AD_Q S TAT 
error halt 



al, id 

al, ql targeted 1, reselect2_chk_lun 
ax, #ERR_SCSI2_REC0NNECT_BAD_ID 
error halt 



al, reconnect^ lun 

al, ql target_lun ], reselect2_f ound 
ax, #ERR_SCSI2_REC0NNECT_BAD_LUN 
error halt 



al, active_cdb 

s aved_acti ve_cdb , al 

active_cdb, qp 

pc, q[ x_reconnect_rtn 



] ; jump to where it disconnected 



program should jmp to last disconnected address 
in case it doesn't, we stop it 

movi.w ax, #ERR__RE SELE CT_S CS 1 2 _RTN 

jmp error_halt 



reselect_scsil : 
mov.b 
mov.b 



al , total_cdb_cnt 
tempq, al 



; search loop begin 
sel_next_scsil_beg : 
movi.b 
jcmpq.b. e 
jmp 

7 

reselectl_srh_id : 
jcmpq.b.e 
jmp 

7 

re9electl_srh_lun : 
mov.b 



al, #QS_DISC 

al, q[ status ], reselectl_srh_id 
get_next_scs i l_queue 



id, q[ target_id ), reselectl_srh_lun 
get_next_sc si l_queue 



al, reconnect lun 
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jcmpq.b.e al, q[ target_lun J, reselectl_found 

get_next_scsil_queue : 

movq.b qp, q[ fwd ) 

mov.b al, tempq 

dec.b al 
mov . b tempq , al 

jcmpi. b.ne al, #ZERO, sel_next__scsil_beg 

; 

; loop expired, cannot find the cdb of disconnected scsil 
mov.b qp, active_cdb 

movi.w ax f #ERR_SCS 1 1 _RE CONNE CT_B AD 

jmp error_halt 



reselect l_f ound : 
mov.b 
mov.b 
mov.b 
movq .w 



al, active_cdb 
saved_active_cdb, al 
active_cdb, qp 
pc, q[ x_reconnect_rtn ] 



; program should not return 
; in case it does, we stop it 

movi.w ax, #ERR_RESELECT_SCSI1_RTO 

jmp error_halt 

; 



ready_cdb_f ound : 
rf lag 
movi.b 
jcmpq.b.ne 
movi.b 
movq.b 
jmp 

; 

ready_cdb_chk_id : 
mov . b 
jcmpq.b.ne 
movi.b 
movq.b 
jmp 



#WTM, #RESET_WTM 
al, #ZERO 

al, q[ targeted ], ready_cdb_chk_id 
al, #QD_BAD_CDB_TRG_ID 
q[ done_stat ], al 
task done 



al , host_scsi_id 

al, q[ target_id ], ready_cdb_chk_busy 
al, #QD_B AD_CDB_TRG_I D 
q[ done_stat ], al 
task done 



al, scsil_busy 

al, q[ target_id ] 

al, #ZERO, ready_cdb_dev_not_busy 



ready_cdb_chk_busy 
mov.b 
andq-.b 
jcmpi .b .e 

is off 

? 

; device has an unfinished command 

; but may accept command if it is SCSI 2 ( support tagged message ) 

; 

mov.b al, scsi2_enable 

andq.b al, q[ target_id ] 

jcmpi. b.ne al, #ZERO, ready_cdb_is_scsi2 



device not busy if bit 



device is scsi 2 if bit ia 



on 



movi.b al, #QS_BUSY 

movq.b q[ status ], al 
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Drop 



idle next cdb 



ready_cdb_is_scsi2 : 
ready_cdb_dev_not Jbusy : 
J 

movi.w ax, #WTM_SEL_TIMEOUT 

mov.w wtm_flag f ax 

♦ 

rf lag ' #WTM,' #TO_250MS 

movq.w id, q[ target_id ] 

sel Init, #ATN 

jtstf.b.bs #SelectDone, selection_completed 

jtstf.b.bs #Reselected, reselect_found 

jtstf.b.bs #Selected, scsi_target_mode 

movi.b al, #QD_SELECT_TIMEOUT 

movq.b q[ done_stat ], al 

jmp task_done 



; watch dog timer ISR vector 



selection_completed : 

rflag #WTM, #RESET_WTM 

movi.b al, #QS_ACTIVE 

movq.b qj status ], al 



set watch dog timer here 



movi.w 
mov.w 
rflag 
jcmpi.b.e 

e 

movi .w 
jmp 

9 

chk_msg_put : 
movq.b 
jtst.b.bc 



; watch dog timer ISR vector 



ax, #ZERO 
wtm__f lag, ax 
#WTM r #TO_10SEC 
ph, #PH_MSG_OUT, chk_msg_out 

ax, #ERRJ*0_ID_MSG_AT_SELECT 
error halt 



al, q[ cntl ] 

al, #eQC_MSG_OUT, identify_msg 



We do not send tagged message if there are specical message(s) to send 
from the global message buffer r thus jmp to setup_cmd_xf er after calling 
mes a a ge_ outride ntif y 



call 
jmp 

# 

identify_msg: 
* 

r 

mov.b 
andq.b 



message_out_identify 
setup_cmd_xfer 



al, scsi2_enable 
al, q[ targeted ] 



jcmpi.b.ne al, #2ERO, identify_scsi2 



identify message scsil 
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rflag #ATN_OFF ; message out last byte, negate attention 

ItlOVi.b al, #( IM_IDENTIFY_MSG | IM_DISC_PRIV ) 

orq.b al, q[ targe t_lun J 

movr.b sb, al ; hardware will send ACK after data sent to bus 

rflag #ACK 

mov.b al, scsil_busy 
orq.b * al, q[ targe't_id j 

mov.b scsil_busy, al ; set current target id busy 

jmp identify_done 



identif y_scsi2 : 
movi.b 
orq.b 
movr.b 
rflag 



al, #( IM_IDENTIFY_MSG | IM_DISC_PRIV ) 

al, q[ targe t_lun ] 
sb, al 
#ACK 



jcmpi.b.e 

movi.w 

jmp 

r 

; tagged queuing 



ph, #PH_MSG_OUT, tagged_queuing_l 
ax, #ERR_NO_ID_MSG_AT_SELECT 
error halt 



? Note: if target 
SS_QUEUE_FULL 



queue is full, we will receive Queue Full stati 



tagged_queuing_l : 
movq . b 
rflag 
jcmpi.b.e 
movi.w 
jmp 

tagged_queuing_2 : 

rflag 
; mov.b 
; movr.b 

movr.b 

rflag 

identify_done: 



either 0x20, 0x21, 0x22 



sb, q[ tag_code ] 
#ACK 

ph, #PH_MSG_OUT, tagged_queuing_2 
ax, #EKR_NOJCD i MSG_AT_SELECT 
error halt 



#ATN_OFF ? negate ATN before last ACK 
al, active_cdb ; use active_cdb as queue tag 
sb, al 

sb, qp 

#ACK 



setup_cmd_xf er : 
movq.w 



ql x_reconnect_rtn ], pc 



cmd_xf er_whi le_not_cmd_out : 

jcmpi.b.e ph, #PB_CMDJDUT, cmd_xf er_outl 

jcmpi.b.e ph, #PH_MSG_IN, cmd_xfer_chk_disc 

jcmpi.b.e ph, #PB_MSG_OUT, cmd_xf er_noop 

jcmpi.b.e ph, #PH_STAT_IN, cmd_xfer^stat_in 

movi.w ax, #ERR_NO_CMD_OUT 

jmp error_halt 
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cmd_xf er_chk_disc : 
call 
jmp 

r 

cmd_xf er_noop : 
call 
jmp 

cmd_xf er_stat_in : 
movi.b 
movq.b 
jmp 

cmd_xf er_outl : 
movq.b 
jtst.b.bc 



chk_di s connect 

cmd xf er while not cmd out 



send_noop 

cnnd_xfer_while_not_cmd_out 



al, #QD_NO_CMD_XFER 
q[ done_stat ], al 
setup_status_x£er 



al, q[ cntl ] 

al, #8QC_REQJ5ENSE, cmd_xf er_out2 



; do request sense 

movi.b ix, #cmd_req_sense 

movi.b al, #CMD_REQ_LEN 

7 

cmd_xf er_req_sense : 

jcmpi.b.e ph, #PH_CMD_OUT, crod_xf erjreq_sensel 

movi.w ax, #ERR_CMD_OUT_INCOMPLETE 

jmp error_halt 

r 

cmd_xf er_req_sensel : 
lodx.b sb 
dec.b al 

jcmpi.b.e al, #ZERO, cmd_xf er_req_sense 

#if o 

; do request sense 
movi.b 
rflag 
jcmpi.b.e 
movi.w 
jmp 

cmdjxf er_sehsel : 
movi.b 
rflag 
jcmpi .b.e 
movi.w 
jmp 

; 

cmd_xf er_sense2 : 
movi.b 
rflag 
j cmpi .b.e 
movi.w 
"imD 

7 

cmd_xf er_sense3 : 
movi.b 
rflag 
jcmpi.b.e 



sb, #CMD_REQ0 
#ACK 

ph, #PH_CMD_OUT, cmd_xfer_sensel 
ax, #ERR_CMD_OUT_INCOMPLETE 
error halt 



sb, #CMD_REQ1 
#ACK 

ph, #PH_CMD_OUT, cmd_xfer_sense2 
ax, #ERR_CMD_OUT_INCOMPLETE 
error halt 



sb, #CMD_REQ2 
#ACK 

ph, #PH_CMD_OUT, cmd_xfer_sense3 
ax, #ERR_CMDJDUT_INCOMPLETE 
error halt 



sb, #CMD_KEQ3 
#ACK 

ph, #PH_CMD_OUT, cmd_xfer_sense4 
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movi.w 



jmp 



ax, #ERR_CMD_OUT_INCOMPLETE 
error halt 



cmdjxf er_sense4 : 
movq.w 
movr.b 
rf lag 
jcmpi.b.e 
movi ; w 
jmp 

; 

cmd_xf er_9ense5 : 
movi.b 
rf lag 
jmp 

#endif 



ax, q[ sense_len ] 

sb, al 

#AGK 

ph, #PH_CMD_OUT, cmd_xfer_sense5 
ax, #ERR_CMDJDUT_INCOMPLETE * 
error halt 



sb, #CMD_REQ5 
#ACK 

setup_data_xfer 



cmd_xf er_out2 : 
movi.b 
movq.w 

7 

cmd_xf er_beg : 

jcmpi.b. 

movi.w 

jmp 

; 

cmd_xf er_out3 : 
movqx.b 
rf lag 
dec.b 
j cropi • b • 



ne 



ix, #cdb ? offset of command buffer 
ax, q[ cdb_len ] ; AX = command len 



ph, #PH_CMD_OUT, cmd_xfer_out3 
ax, #ERR_CMD_OUT_INCOMPLETE 
error halt 



sb, q[ ix ] 

#ACK 

al 

al, #ZERO, cmd_xfer_beg 



if both QCJDATA_IN & QC_DATA_OUT bit set, then no data transfer is assumed 
date: 5/23/93 



movi.b 
andq.b 
jcmpi.b.e 
jmp 

9 

cmd_xf er_no_ data : 
movq.b 
jtst.b.bc 

movi .w 
jmp 



al, #( QC__DATA_IN | QC_DATA_OUT ) 
al, q[ cntl ] 

al, #( QC_DATA_IN | QC_DATA_OOT ), cmd_xf er_no_data 
cmd_xfer_tst_sg_list 



al, q[ cntl ] 

al, #eQC_SG_LIST, setup_status_req_wait 

ax, #ERR_SGJCiIST_NOJ>ATA_XFER 
error halt 



? 

cmd_xf er_ts t_sg_list : 

movq.b al, q[ cntl J 

jtst.b.bs al, #8QC_SG_LIST, cmd_xf er_sg_list 



not sg_list 
movq.w 
movq.w 
movq.w 
movq.w 
movq.w 
movq.w 



ax, q[ data_addrO ] 

q[ x_saved_data_addrO ], ax 

ax, q[ data_addrl ] 

q[ x_saved_data_addrl ], ax 

ax, q[ data_cntO ] 

q[ x_saved_data_cntO ], ax 
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movq.w ax, q[ data_cntl ] 

movq.w q[ x_saved_data_cntl ], ax 

jmp setup_data_xfer 



cmd_xf er_sg_list : 



; setup transfer address low and high word 

movq.b qp, q[ x_saved_sg_list_qp J ; change qp to sg_list 

e 

movq.b al, q[ sg_entry_cnt J ; al to be saved into sgJList head 

movi.b ix, #sg_list0 ; 

movqx.w daO, q[ ix ] ; ix=ix+2 

movqx.w dal, q[ ix ] • ix=ix+2 

setup ix initial value 

if we are disconneted, after reconnection we restore dma register 
from saved area, we do not use ix to move entry, 
so ix should always point to next entry 

setup q[ x_saved_sg_entry_cnt ] , and q[ x_saved_sg_entry_index ] 
mov.b qp, active_cdb 

9 

; setup transfer byte count, low and high words 
movq.w dcO, q[ sg_f irst_xf er_cnt ] 

movi.w del, #ZERO ; high word of transfer count should be 2ero 

movq.b q[ x__saved_sg_entry_cnt J, al 

movq.b q[ x_saved_sg_index ], ix 



setup_data_xf er : 

movq.w q[ x_reconnect_rtn ], pc 

} 

data_xf er_chk_data_in : 

jcmpi.b.e ph, #PH_DATA_IN, data_xf er_chk_dir_in 
jmp data_xfer_chk_data_out 

7 

data_xf er_chk_dir_in : 

movq.b al, q[ cntl ] 

jtst.b.bc al, #8QC_DATA_OUT, data_xf er_beg 

. 

9 

movi.w ax, #ERR_INVALID_DATA_IN_PHASE 

jmp error_halt 

data_xf er_chk_data_out : 

jcmpi.b.e ph, #PH_DATA_OUT, data_xf er_chk_dir_out 
jmp data_xf er_chk_msg_in ~ 

r 

data_xf er_chk_dir_out : 

movq.b al, qf cntl ] 

jtst.b.bc al, #6QC_DATA_IN, data_xf er_beg 

r 

movi.w ax, #ERR_INVALID_DATA_OUT_PHASE 

jmp error halt 
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data_xf er_chk_msg_in : 

jcmpi.b.ne ph, #PH_MSG_IN, data_xf er_chk_msg_out 

call chk_di9connect 

jmp data_xfer_chk_data_in 

7 

data_xfer_chk_msg_out : 

jcmpi.b.ne ph, #PH_MSG_OUT, data_xf er_chk_stat in 

call send_noop 

jmp data_xfer_chk_data_in 

9 

data_xf er_chk_stat_in : 

jcmpi.b.e ph, #PH_STAT_IN, data_xfer,err_phase 
movi.w ax, #ERR_NO_DATA_PHASE 

jmp error_halt 

? 

data_xf er__errj>hase : 

movi.b al, #QD_NO_DATA_XFER 

movq.b q[ done_stat ], al 

jmp se tup_s t atu s_xf er 

7 

data_xf er_beg : 

movq.b al, q[ cntl ] 

jtst.b.bc al, #8QC_REQ_SENSE, data^xf er_tst_sg_list 
Request Sense data_xfer 

movq.w. daO, q[ sense_addrO ] 

movq.w dal, q[ sense_addrl ] 

movq.w dcO, q[ sense_len ] 

movi.w del, #ZERO 

movi.b .al, #QS_DATA_XFER 

movq.b q[ status ] , al r 
dma 

jtstf.b.bs #DcZero, setup_statusjreq_wait 

movi.w ax, #ERR_SENSE_XFER_INCOMPLETE 

jmp error_halt 

* 

9 

data_xf er_tst_sg_list : 

movq.b al, q[ cntl ] 

jtst.b.bc al, #§QC_SG_LIST, data_xf er_not_sg_list 
7 — — — — — — ——>———-.— — 

; prepre data transfer for sg list 

movi.b al, #QS_DATA_XFER 

movq.b q[ status ], al 



SG LIST DATA XFER BEGIN 



data_xf er_sg_list i 
dma 



dma tranfer count not expired 
the disconnected or error exit 



jtstf.b.bc #DcZero, dma_xf er_dcjiot_zero 
movq.b al, q[ x_saved_sg_entry_cnt ] 
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dec.b al 

movq.b q[ x^saved_sg_entry_cnt ] , al ; don't destory al til 

jcmpi.b.ne al, #ZERO, data_xf er_f etch_next 

9 

; al is remain of sg entry count of 
? al is zero, test end of list 
• 

movq.b al, q[ sg_list_fwd_qp ] ; al is used as next qp 111 

; do not destroy it ■ 
jcmpi.b^e al, #QLINK_TAIL, setup_status_req_wait ; exit from here M 

the entry count expired on this sg_list queue 
we shall change q[ x_saved_sg_list_qp J to next sg_list queue 

Note: ix will be incremented when we move next entry address 
into dma register by using index move instruction 

movi.b ix f #sg_list0 ? set ix to beginning of first sg_list 
movq.b q[ x_saved_sg_list_qp ] , al ; set current working 

sg_list_qp 
J 

movq.b qp, q( x_saved_sg_list_qp ] 

movq.b al, q[ sg_entry_cnt ] 

mov.b qp, active_cdb 

movq.b q[ x_saved_sg_entry_cnt ] , al 

initialize q[ sg_list_fwd_qp J 

movq.b qp, q[ x_saved_sg_list_qp ] 

movq.b al, q{ fwd J 

mov.b qp, active_cdb 

movq.b q[ sg_list_fwd_qp ] , al 

fetch next entry address of sg_list 

data_xf er_f etch_next : 

in this sg_list loop , the ix is saved berfore incremented 

movq.b qp, q[ x_savedj3g_list_qp ] 

movqx.w daO, q[ ix ] ; ix=ix+2 

movqx.w dal, q[ ix ] ; ix=ix+2 

mov.b qp, active_cdb ; restore qp 

** the al should contains q[ sg_entry_cnt ] 
if we come to here al is non-zero value 

jcmpi.b.ne al, #0NE, data_xf er_jK>t_last_sg_list 

movq.b al, q[ sg_list_fwd_qp ] 

jcmpi.b.e al, #QLINK_TAIL , data_xf er_last_sg_list 

§ 

; more than one sg entry remain 
? 

data_xf er_not_last_sg_list : 

movq.w dcO, q[ sg_page_size ] 

movi.w del, #ZERO 

jmp data_xfer_sg_list 
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The last entry of sg_list 

data_xf er_last_sg_list : 

movq.w dcO, q[ sg_la9t_xf er_cnt ] 

movi.w del, #ZERO 

jmp data_xfer_sg_list 

prepare data transfer for non-sg_list 

data_xfer_not_sg_list: 

movq.w daO, q[ x_saved_data_addrO ] 

movq.w dal, q[ x_saved_data_addrl ] 

movq.w dcO, qj x_saved_data_cntO ] 

movq.w del, q[ x_saved_data_cntl ] 

movi.b al, #QS_DATA_XFER 

movq.b q[ status J, al 

dma 

j tstf .b .be #DcZero, dma_xf er__dc_not_zero 
jmp setup_status_req_wait 

DMA completed , transfer count non-zero 

dma_xf er_dc_not_zero : 

; date: 5-26-93 

jtstf.b.bc #ParityError, dc_not_zero_wait_status;_in 

rflag #PARITY 

movi.W .ax, #ERR_PARITY__DATA_IN 

jmp error_halt 

» 

dc_not_zero_wait_status_in : 

jcmpi.b.e ph, #PH_STAT_IN, dc_not_zero_stat_in 
jcmpi.b.ne ph, #PH_MSG_IN, dc_not_zero_waitjmsg_put 
call chk_disconnect 
jmp dc_not_zero_wait_status_in 

9 

dc_not_zero_wait_msg_out : 

jcmpi.b.ne ph, #PH_MSG_OOT, dc_not_zero_jio_stat_in 

call send_noop 

jmp dc_not_zero_wait_status_in 

r 

dc_not_zero_no_stat_in : 

movi.w ax, #ERR_NO_STAT_IN 

jmp errorjialt 

#if 0 

movi.b al, #( QCJDATA_IN | QC_DATA_OUT ) 

andq.b al, q[ cntl ] 

jcmpi.b.ne al, #ZERO, dc_not_zero_err ; check transfer count error 
jmp setup_status_xfer ; ignore tranfer count incomplete 

#endif 

; dc_not_zero_err : 
dc not zero stat in: 
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movi.b al, #QD_DATA_XFER_UNDER_RUN 

movq.b q[ doners tat J, al 

jmp setup_s t a tu s_xf er 



setup_status_req_wait: .... 

movq.w q[ x_reconnect_rtn J, pc 

; date: 5-26-93 

jtstf.b.bc #ParityError, wait_status_in 

rflag #PARITY 

movi.w ax, #ERR_PARITY_DATA_IN 

jmp error_ halt 

wait status phase or disconnection 

wait_status_in: 

jcmpi.b.e ph, #PH_STAT_IN, setup_status_xf er 

jcmpi.b.ne ph, #PH_MSG__IN, wait_msg_out 

call chk_disconnect 

jmp wait_status_in 

J 

wait_msg_out: 

jcmpi.b.ne ph, #PH_MSG_OUT, err_no_stat_in 

call send_noop 

jmp wait_status_in 

if we come to here, target can only be in either of following three phases 

1. PH_DATA_OUT: data out over run, target still in data out phase 

2. PH_DATA_IN: data in over run, target still in data in phase 

3. PH_CMD_OUT: 

err_no_stat_in : 

movi.w ax, #ERR_NO_STAT_IN 

jmp error_halt 



setup_status_xf er : 
movq.b 
rflag 

9 

jcmpi.b.e 

movi.w 

jmp 

. 

status_chk_done : 
movr.b 
rflag 
movq.b 
jcmpi.b.e 

jcmpi.b.ne 
movi.w 



q[ scsi_ stat J , 
#ACK 



sb ; get status 



ph, #PH_MSG_IN, status_chk_done 
ax, #ERR_NO_CMD_CMPL_MSG 
error halt 



al, sb 
#ACK 

q[ scsi_msg J, al 

al, #MS_CMD_DONE , status_done 



al, #Ml_LINK_CMD_DONE, status_chk_link_wf lag 
al, #ERR_LINK_CMD__DONE 
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jmp error_halt 

« 

status_chk_link_wf lag : 

jcmpi.b.ne al, #M1_LINK__CMD_D0NE_WFLAG , status_err_scsi_jnsg 

movi.w ax, #ERR_L I NK_CMD_DONE_WFLAG 

jmp error_halt 

7 

statu s_err_s c s i_ms g s 

movi.w ax/ #ERR_CMD_DONE_MSG 

jmp error_halt 

9 

status_done : 

waitfree 

movi.b al f #SS_GOOD 

jcmpq.b.e al, q[ scsi_stat ], task_done 

7 

movi.b al, #SS_TARGET_BUSY 

jcmpq.b.ne al, q[ scsi_stat ], status_chk 

9 

movi.b al, #QS_BUSY 

movq.b q[ status ], al 

9 

status_chk: 

movi.b al, #SS_CHK_CONDITION 

jcmpq.b.ne al, q[ scsi_stat ], status_bad 

7 

movq.b al, q[ cntl ] 

jtst.b.bs al, #6QC_REQ_SENSE, status_cannot_get_sense 

do request snese, original q[ cntl ] destroyed 
date: 5-26-93 

movi.b al, #( QC_REQ_SENSE | QC_DATA__IN > ; force do request sense 

movq.b q[ cntl J, al 

movi.b al, #QS_READY 

movq.b q[ status ] , al 

jmp ready_cdb_found 

statu s_cannot_get_sense : 

movi.w ax, #QD_CAN_NOT_GET_SENSE 

jmp task_done 

9 

status_bad: 

movi.b al, #QD_BAD_SCSI_STATUS 

movq.b q[ done_stat ], al 



task_done: 

rflag #WTM, #RESET_WTM 

mov.b al, scsil_busy 

andq.b al, q[ target_id ] 
jcmpi.b.e al, #ZERO, task_done_unlink_q ; if not scsi 1, goto 
task_done_x 



is scsi 1, clear scsil_busy 
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mov.b al, scsil_busy 

xorq.b al, q[ target_id ] 

mov.b scsil_busy, al 

7 

task_done_unlink_q : 

mov.b al, total_cdb_cnt 



jcmpi.b.e al, #ONE, task_done_unlink_done 

; current qp changed to q[bwd] 



movq.b 
movq.b 
movq.b 



al, q[ fwd ] 
qp, q[ bwd ] 
q[ fwd ], al 



mov.b qp, active_cdb ; restore active_cdb 

movq.b al, q[ bwd ] 

movq.b qp, q[ fwd ] ; current qp changed to q[fwd] 

movq.b q[ bwd ], al 

7 

mov.b qp, active_cdb ; restore active_cdb 

? 

task_done__unlink_done : 

movq.b al, q[ fwd 1 

mov.b next_active_cdb, al 

mo v . b a 1 , r i s c_done_ next 

movq.b q[ bwd ], al 

7 

movq.b al, q[ cntl J 

jtst.b.bs al, #SQC_SG_I*IST, task_done_sg_list 

7 

; not sg_list, mark q[ fwd J as tail 
• 

9 

movi.b al, #QLINK_TAIL ; mark the queue as end of queue 

movq.b q[ fwd ], al 

jmp task_done_set_risc_done_next 

mark q[ fwd ] of sg_list's cdb to its sg_list 

task_done_sg_list : 

movq.b al, q[ sg_list__qp ] 

movq.b . q[ fwd J , al ; let active_cdb->q_fwd 

>qx_sg_li9t_QP 



active cdb- 



task_done_ set_risc_ done_next : 

movi.b al, #QS_DONE 

movq.b q[ status ] , al 

mov.b qp, risc_done_next ; qp changed to risc — done_next 

movq.b al, q( fwd ] 

jcmpi.b.e al, #QLINK_TAII>, task_done_link_done_list 

r 

; error I tail of done list is not #QLINK_TAIL 
7 

movi.w ax, #ERR_DONE_LINK_CORHUPTED 

jrop error_halt 

link q[ fwd J of done list tail to active_cdb 

task_done_link_done_list : 

mov.b al, active_cdb 

movq.b q[ fwd ], al 
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mov.b risc_done_next, al 

mov.b qp, active_cdb 

search end of list, both sg_lis and non-sg_list 
then set risc_next_done to end of active queue 

done_set_r isc_done_next : 

movq.b al, q[ fwd ] 

"jcmpi.b.e* al, #QLINK__TAIIt , task_done_tail_found 
movq.b qp, q[ fwd ] 

mov.b risc_done_next, qp ; do not put QLINKJTAIL into it 

jmp done_set_risc_done_jiext 

r 

task_done_tail_f ound : 

sint ; set interrupt to host 

; 

mov.b al, total_cdb_cnt 

dec.b al 

mov . b total_cdb_cnt , al 

when the total_cdb__cnt is zero, not even active_cdb is valid ! I ! 

jcmpi.b.e al, #ZERO, idle_no_cdb ; total_cdb_cnt = 0 

jcmpi.b.e al, #ONE, get_next_active_cdb 

mov.b al, saved_active_cdb 

jcmpi.b.e al, #QLINKJTAIL, get_next_active_cdb 

reconnection occured in doing saved__active_cdb, redo the saved_active_cdb 

mov.b qp, saved_active_cdb 

movi.b al, #QS_READY 

jcmpq.b.ne al, q[ status ], get_next_active_cdb 

mov.b active_cdb, qp 

jmp idle_next_cdb 

there is not reconnection happened in processing this cdb 

get_next_active_cdb : 

mov.b qp, next_active_cdb 

mov.b active_cdb, qp 

jmp idle_next_cdb 



chk_dis connect : 

jcmpi.b.ne ph, #PH_MSG_JEN, check_disc_end 

date: 5-30-93 

movr.b al, sb 

rflag #ACK 

jcmpi.b.ne al, #M1_SAVE_DATA_PTR, checkjiisc_msg 
date: 6/15/93 mark out checking q[ status ] 
movq.b al, q[ status ] 

jcmpi.b.ne al, #QS_DATA_XFER, save_data_ptr_err_status 
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9 — — — _ — 

; save current dma register 

movq.w q[ x_saved_data_addrO ], daO 

movq.w q[ x__saved_data_addrl ] , dal 

movq.w q[ x_saved_data_cntO J, dcO. 

movq.w q[ x_saved_data_cntl J, del 

movq.b q[ x_saved_sg_index ] , ix ; save the ix for disconnection 

r 

jmp chk_disconnect 

• **— — — — ____ — — — _ 

; currently this error is not processed in 

9 

save_data_ptr_err_status : 

movi.w ax, #ERR_SAVE_DATA_PTR_STATUS 

jmp errorjialt 

9 *" — *~ — — — — — — _ — — — — — ~ 

check_disc_msg: 

jcmpi.b.ne al, #Ml_DISCONNECT, check_disc_noop 

disconnect message 

waitfree 

movi.b al, #QS_DISC 

movq.b q[ status ], al 

jmp id 1 e_ nex t_cdb 

check_disc_noop : 

jcmpi.b.ne al, #Ml_NO_OP, check_disc_restore_j?trs 

? 

; do no thing when you get a noop message ! ! 

9 

jmp chk_dis connect 

• ~ - 

check_disc__restore__ptrs : 

jcmpi.b.ne al, #M 1_RE STORE_PTRS , check_disc_ext_msg 

; handle Restore Pointer Message 

movq.b al, q[ status ] 

jcmpi.b.ne al, #QS_DISC, res_data_ptr_ err_status 

9 m 

movq.w daO, q[ x_saved_data_addrO J 

movq.w dal, q[ x_saved_data_addrl ] 

movq.w dcO, q[ x_saved_data_cntO 1 

movq.w del, q[ x_saved_data_cntl ] 

movq.b ix, q( x_saved_sg_index ] ; restore ix from disconnection 

jmp chk_disconnect 

9 

res_data_ptr_err_status : 

movi.w ax, #ERR_RE S_D AT A_P TR_S TATUS 

jmp error_halt 

; 

check_disc_ext_msg : 

9 

j cmpi . b . ne al , #MS_EXTEND , check_disc_more 

mov.b msg_in_buf f er, al 

call message_in_01 
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returned from message_in subroutine 
see if host want us to send message out 

movq.b al, q[ cntl ] 

jtst.b.bc al, # @ QC_MSG_pUT , check_disc_no_more_msg 

rf lag #ATN_ON ; we have message to send 

rf lag #ACK ; we ACK the last bytes 

jcmpi,b.e ph, #PH_MSG^OUT, check_disc_msg_out 

raise attension failed ! 

movi.w ax, #ERRJRAI SE_ATN_FAILED_0 1 

jmp error_halt 

host want us to send message out as result of last message in 

check__disc jmsg_out : 

call message_out_beg 
jmp chk_disconnect 

? 

check_disc_no_more_insg 2 

rf lag #ACK ; we ACK the last byte 

? 

jmp chk_disconnect 

} 

checkjiisc^jnore : 

. ******************************* 

; for implement more check 
; not implemented yet ill 
7 

jmp chk_disconnect 

f 

check_disc_end : 
~ ret 



send_noop : 

rf lag #ATN_OFF ; reset attention before last ACK 

movi.b al, #Ml_NO_OP 

movr.b sb r al ; 

rflag #ACK 

ret 



error_halt: 

mov.w halt_code, ax 

halt #INT 

mov.b qp, active_cdb ; restore active_cdb 

; when error halt, total_cdb_cnt cannot be zero, 
; so we jump back to idle_next_cdb 

jmp idle_next_cdb 
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watch dog- timer time out ISR 



wtm isr: 



mov.b 
mov.b 
jtst.b.bs 

movi.b 
movq.b 
jmp 



; restore active cdb 



qp, active_cdb 
al, wtm_f lag 
al, # € WTM_SE L__T I ME OUT , sel_timeout_isr 

al/ #QD_WTM_TIME OUT 
ql done_stat ] , al 
task done 



selection time out ISR 



sel_timeout_isr : 
movi.b 
movq.b 
jmp 



al, #QD_SELECT_TIME OUT 
q[ done_stat ], al 
task, done 



decrement q[ timeout_cnt J until zero 



dec_timeout_ cnt : 
mov.b 
mov.b 

dec_timeout_beg : 
movi.b 
jcmpq.b.e 
jmp 

9 

dec_timeout_disc : 
movq.b 
jcmpi.b.ne 
jmp 

7 

dec_timeout_chk : 
movq.b 
jcmpi.b.ne 
movq.b 
jcmpi.b.ne 
jmp 



al, total_cdb_cnt 
teropq, al 



al, #QS_DISC 

al, q[ status ], dec_timeout_disc 
dec timeout next 



al, q[ timeout_chk ] 

al, #ZERO, dec_timeout_chk 

dec timeout next 



al, q[ timeout_cntl ] 

al, #ZERO, dec_msb_not_zero 

al, q[ timeout_cntO ] 

al, #ZERO, dec_msb_zero_lsb_not_zero 

dec_timeout_ next ; already zero i ! 



MSB is not zero, LSB unknown at this time 



dec_msb_not_ zero ; 
movq.b 
dec.b 
movq.b 
jcmpi.b.e 
jmp 

7 

dec_msb_also : 
movq . b 
dec.b 
movq.b 



al, q[ timeout_cntO ] 
al 

q[ timeout_cntO ] , al 
al, #0xFF, dec_msb_also 
dec_timeout next 



al, q[ timeout_cntl ] 
al 

q| timeout_cntl ], al 



WO 95/06286 



PCT/US94/09415 



- 76 - 



3mp 



dec_timeout_next 



dec_msb_zero_lsb_not_zero : 

movq.b al, q[ timeout_cntO ] 

dec.b al 

movq.b q[ timeout_cntO J, al 



dec__timeout_next i 
mov . b 
dec.b 
xnov.b 
jcmpi.b.e 
movq.b 
jmp 

dec_timeout_done : 
mov.b 
ret 



al, tempq 
al 

tempq, al / 
al, #ZERO, dec_timeout_done 
qp, q[ fwd ] 
dec_timeout_beg 



qp, active_cdb 



; 



message_out_identif y : 

movi.b al, #( IM_IDENTIFY_MSG ) 

al, q{ target_lun ] 

sb, al ; hardware will send ACK after data sent to bus 
#ACK 



orq.b 
mo vr . b 
rflag 



jcmpi.b.e ph, #PH_MSG_OUT, message_out_beg 



movi.w 
jmp 



ax, #EKR_NO_I D_MS G_AT_SELE CT 
.error halt 



; the entry point is called by checkjiisconnect to send out message" 
? 

message_put_beg : 

mov.b al, msg_in_buf fer 

jcmpi.b.ne al, #MS_EXTEND, message_out_not_ext ; do not destroy al 
send extended message, the al contains first byte of message 



movi.b 
lodx.b 
rflag 
mov.b 



ix, #2ERO 

sb ; extended message first byte 

#ACK 

al, ext_msg_len ; extended message length 



message_ext_loop_beg : 

jcmpi.b.e ph, #PH_MSG_OUT, message_ext_loopl 
movi.w ax, #ERR__NO_ID_MSG_AT_SELECT 

jmp error_halt 

7 

message_ext_loopl : 

lodx.b sb 
rflag #ACK 

# 

dec.b al 

7 

jcmpi.b.ne al, #ZERO, message_ext_loop_beg 
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; message_ext_loop__end: 

; 

jcmpi.b.e ph, #PH_MSG__OUT, message_ext_last_byte 

movi.w ax, #ERR__NO_ID_MSG_AT_SELECT 

jmp errorjialt 

9 

message_ext_last_byte : 

• -rflag - #ATN_OFF • 
lodx.b sb 

rflag #ACK 

; 

jcmpi.b.e ph, #PH_MSG_IN, get_ext_msg_in 

movi.w ax, #HALT_EXT_MSG 

mov . w halt_code , ax 

* halt #INT 

jmp t e s t_ext_ms g_done 

get_ext_msg_in : 

call message_in 

9 

te s t_ext_ms g_done : 
t 

; if the QC_MSG_OUT flag is still on, send more message 

■9 

movg.b al, q[ cntl ] 

jtst.b.bs al, #6QC_MSG_OUT, preparers g_out_a gain 

rflag #ACK 
ret 

prepare_msg_out_again : 

We will raise attention again to send out more message 

sel Init, #ATN 

rflag #ATN_ON 

rflag #ACK 

jcmpi.b.e ph, #PB_MSG_OUT, message_out_beg 

movi.w ax, #ERR_RAISE_ATN_FAIL£D_02 

jmp error_halt 



; 

; send message, the al contains first byte of message 

9 

message_out_not_ext : 

; 

rflag #ATN_OFF 
movr.b sb, al 

rflag #ACK 
ret 



this subroutine handles: 
1. extended message 
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Note: the subroutine does not reset ACK after last byte is sent 

message_in: 

movr.b al, sb 

rflag #ACK 

9 

mov . b ms g_in_buf f er , al 

jcmpi.b.ne al, #MS_ EXTEND , not_ext_message__in 

this entry point is called by check_discconect when extended 
message is received, 

message_in_01 : 

jcmpi.b.e ph, #PH_MSG_IN, get_ext_msg_in_len 
movi . w ax , #ERR_EXT_MSG_IN_ERRORl 

jmp error_halt 

9 

get_ext_msg_in_len : 

movr.b al, sb 

rflag #ACK 

mov.b , ext_msg_len , al 

movi.b ix, #2 ? start from 

9 

get_ext_msg_in_loop_beg : 

jcmpi.b.e ph, #PH_MSG_IN, get_ext_msg_in_loopl 
movi.w ax, #ERR_EXT_MSG_IN_ERROR2 

jmp error_halt 

J 

get_ext_msg_in_loopl : 
dec.b • al 

jcmpi.b.e al, #ZERO, get_ext_msg_in_loop_end 

7 

stox.b sb ; store [ix] to sb, ix=ix+l 

rflag #ACK 

jmp get_ext_msg_in_loop_beg 

9 

get_ext_msg w in_loop_end : 

The last byte is not received yet 

We will not reset ack after it is sent as to prevent target from changing 
phase 

stox.b sb ; store the last byte of message in 

movi.w ax, #HALTJEXT_MS G 

mov.w halt_code, ax 

halt #INT ; waiting for host to restart CPU 
ret 



not_extjnessage_in : 

movi»W ax, #ERR_UNKNOWN_MSG_IN_01 

jmp error_halt 

ret 
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END ; End Of File 
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APPENDIX IV 



- This is synthesize model for SEAL register decode logic 



-library WORK; 

- use WORKLsys_pkg.all; 

entity reg_dec is 
port( 

cs : in vlbit; 
master : in vlbit; 

adr : in vlbit_vector (3 downto 2); 
ben : in vlbhjvector (3 downto 2); 
benO: in vlbit; 
reg_bankl : in vlbit; 

adrl: out vlbit; 
adrO : out vlbit; 

risc_cs : out vlbit; — rise register address 

confrom_cs : out vlbit; - configuration register and EEprom 

h_pc_cs : out vlbit; 

h_ofs_cs : out vlbit; 

lm_cs : out vlbit; 

contrives rout vlbit; 

stat_int_.es : out vlbit; 

h_cnt_cs : out vlbit; - transfer pointer and counter 

hjd_adr : out vlbit; 

sc_dat_adr : out vlbit; 

sc_cd_adr : out vlbit; 

h_fifo_.es : out vlbit; 

flg_adr: out vlbit 

); 

end reg_dec; 



architecture BEHAVIORAL of reg_dec is 

signal adiOj : vlbit; 
signal adrl J : vlbit; 
signal confrom_i : vlbit; 

begin 

adrl_i<=NOTben(3); 
adiO_i <= ben(2) and benO; 
adrO <= adrO_i; 
adrl <= adrl J; 
confrom_.es <= confromj; 

confrom_i <= cs and NOT master and NOT reg_bankl and NOT adr(3); 
lm_.es <= cs and NOT master and NOT reg_bankl and adr(3); 
hjpc_cs <=cs and NOT master and NOT reg_bankl and adr(3) and adr(2) 
and NOT adrl J; 

h_ofs_cs <= cs and NOT master and NOT regj>ankl and adr(3) and NOT adr(2) 

and adrl J and adrOj; 
stat_int_.es <= cs and NOT master and NOT reg_bankl and adr(3) and NOT adr(2) 
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and NOT adrlj and adiOJ; 
contrives <= cs and NOT master and adr(3)and adr(2) 

and adrl_iand adiO_i; 

risc_cs <= cs and NOT master and reg_bankl and NOT adr(3) and NOT adr(2); 
h_fifo_cs <= cs and NOT master and reg_bankl and NOT adr(3) and adr(2); 
h_id_adr <= cs and NOT master and reg_bankl and NOT adr(3) and adr(2) 

and NOT adrl J and adiOJ; 
flg_adr <= cs and NOT master and reg_bankl and NOT adr(3) and adr(2) 

and adrl J and adiO_i; 
h_cnt_cs <= cs and NOT master and reg_bankl and adr(3); 
sc_ctl_adr <= cs and NOT master and regjrankl and adr(3) and NOT adr(2) 

and NOT adrl J and adiOJ; 
sc_dat_adr <= cs and NOT master and regj>ankl and adr(3) and NOT adr(2) 

and adrl J and adiO_i; 



end BEHAVIORAL; 
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- This is behavioral model for top level Local Memory 



-library WORK; 

-use WORK.sys_pkg.all; 

entity lm_ctl is 
port( 

- RISC block interface 

RCMREQ: in vlbit; — level, reset by RCMDONE 

RCMDONE : out vlbit; - pulse, local memory to RISC acknowlodge 

RWRENr in vlbit; - 1 = write, 0= read 

WORD : in vlbit; - 1 = word, 0 = byte 

RMEM_DIN : in vlbit_vector(15 downto 0); 

RMEM_JDO_H : out vlbit_vector(15 downto 8); 

RMEMJDOJL : out vlbit_vector(7 downto 0); 

RMEM_ADR : in vlbit_vector(14 downto 0); 

- VESA / IS A block interface 

- VESA is always word operation 

HJNJBUS : out vIbit_vector(15 downto 0); 

H_OUT_BUS : in vlbit_vector(15 downto 0); 

HOSTJX5NE rout vlbit; 

LRAM_CS: in vlbit; 

H_ADR : in vlbit_vector(2 downto 0); 

H_WR : in vlbit; 

H_RD : in vlbit; 

- External memory interface 

LMEM_DIN_H : out vIbit_vector(15 downto 8); 
LMEM_DIN_L : out vlbit_vectorC7 downto 0); 
LMEM_DOUT : in vlbit_vector(15 downto 0); 
LMEM_ADR : out vlbit_vectar(14 downto 1); 
CS0N : out vlbit; 
CS1N rout vlbit; 
HSJX)NE rout vlbit; 
LMEM.WRN rout vlbit; 

- Genernal signal 

MEM_WATT : in vlbit_vector(l downto 0); 
MEM8 r in vlbit; 
RST r in vlbit; 
CLK: in vlbit 

); 

end lm_ctl; 

architecture BEHAVIORAL of lm_cd is 
signal hr_state : vIbit_vector(l downto 0); 
signal lp_state r vlbit_vector(2 downto 0); 
signal lmc_state r v!bitjvector(2 downto 0); 

constant lp_st0v: vlbit_vector(2 downto 0) := b w 000"; 
constant lp„stlv: vlbit_vector(2 downto 0) := b"001 n ; 
constant lp_st2vr vlbit_vector(2 downto 0) := b w 010"; 
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constant lp_st3v: vlbit_vector(2 downto 0) := b"0ir; 
constant lp_st4v: vlbit_vector(2 downto 0) := b"100"; 
constant lp_st5v: vlbit_vector(2 downto 0) := blOl"; 

* -constant lp_st6v: vlbit_vector(2 downto 0) := b'llO"; 

constant idler : vlbit_vector(2 downto 0) := b w 000 n ; 
constant cy_10v : vibit_vector(2 downto 0) := b°00r; 
constant cy_l 1 v : vlbit_vector(2 downto 0) := b"010"; 
constant cy_12v : vlbit_vector(2 downto 0) := b"01 1"; 
constant cy_20v : vlbit_vector(2 downto 0) := b w 100 M ; 
constant cy_21v : vlbitjvector(2 downto 0) := b"10r; 
constant lastv : vIbit_vector(2 downto 0) := b n ll<T; 

constant hr_st0v: vlbit_vector(l downto 0) r= b tt 00"; 
constant hr_stlv: vlbit_vector(l downto 0) := b"0r; 
constant hr_st2v: vlbit_vector(l downto 0) := b"10"; 
constant hr_st3v: vlbit_vector(l downto 0) := b"ll"; 

constant lp_st0: integer := 0; 
constant lp_stl: integer := 1; 
constant lp__st2: integer := 2; 
constant lp_st3: integer := 3; 
constant lp_st4: integer := 4; 
constant lp_jst5: integer := 5; 
-constant lp_st6: integer := 6; 

constant idle : integer := 0; 
constant cy_10 : integer := 1; 
constant cy_l 1 : integer := 2; 
constant cy_12 : integer := 3; 
constant cy_20 : integer := 4; 
constant cy_21 : integer := 5; 
constant last : integer := 6; 

constant hrjstO: integer := 0; 
constant hrjstl: integer := 1; 
constant hr_st2: integer := 2; 
constant hr_st3: integer := 3; 

signal hmreq : vlbit; 
signal hwren : vlbit; 
signal risc_gnt : vlbit; 

* signal host__gnt : vlbit; 

signal lat.r_.dat0 : vlbit; 

signal HWR_LMEM_DAT : vlbit; 
signal HWRJLMEM_ADR : vlbit; 
signal HRD_LMEM_DAT : vlbit; 
signal HRD_LMEM_DATjL : vlbit; 
signal HRD JLMEMJD ATJK : vlbit; 
signal HRDJ-MEM_ADRJL : vlbit; 
signal HRDJLMEM_ADRH : vlbit; 
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signal hjmemjeg : vlbit_vector(15 downto 0); 
signal h_lmem_regji_in : vlbit_vector(7 downto 0); 
signal h_lmem_reg_ljn : vlbit_vector(7 downto 0); 
signal hjmem_adrh : vlbitjvector(15 downto 8); 
signal hjtaiem_adii : vlbit_vector(7 downto 0); 
— signal hjbnem_adr_in : vlbit_vector(15 downto 0); 
signal r_lmem_reg : vlbit_vector(7 downto 0); 
signal res_17 : vlbit_ld(16 downto 0); 

signal lmem_proc : vlbit; 
signal wr_en : vlbit; 
signal latch_data : vlbit; 
signal rsrproc : vlbit; 
signal start_proc : vlbit; 
signal done : vlbit; 
signal wr_enjnc_adr : vlbit; 
signal rd_en Jnc_adr : vlbit; 

signal cycle_l : vlbit; 
signal cycle_2 : vlbit; 

signal csl_meml6 : vlbit; 
signal csl_mem8 : vlbit; 
signal incjhjadr : vlbit; 
signal inc_h_adr_clkh : vlbit; 
signal incjh_adr_clkl : vlbit; 
signal lmem_wrjidatj : vlbit; 
signal lmem_wrjidat.j_clk : vlbit; 
signal lmem8_wr_hdat_Ji : vlbit; 
signal lmem8_wr_hdat_h_clk : vlbit; 
signal taeml6_wr_hdat_h : vlbit; 
signal lmeml6_wr_hdat_h_clk : vlbit; 

begin 



control : block 
begin 

LMEM_WRN <= not ((host_gnt and HWREN and wr_en) or 
(rise _gnt and RWREN and wr_en)) ; 

CS0N <= not (host_gnt or 

(risc_gnt and not (cycle_l and RMEM_ADR(0) and (not WORD) and not MEM8 ))) ; 

csl_meml6<= host_gnt or 

(risc_gnt and cyclej and WORD ) or 

(risc_gnt and cycle_l and not WORD and RMEM_ADR(0)); 
csl_mem8 «c= (host_gnt and cycle_2 ) or 

(risc_gnt and cycle_2 ) or 

(risc_gnt and cycle_l and not WORD and RMEMADR(0)); 
CS1N <= (not MEMS and not csl_meml6) or 
(MEM8 and csl_mem8); 
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-CS1 = lmein_adr(0) when MEM8=1 

~CSl = lmem_csl when MEM8=0 or MEM16=1 

lat_r_dat0 <= risc_gnt and (not RWREN) and latch_data and cycle_l and WORD and not MEM8; 
RCMDONE <- risc_gnt and done; 
- RMOE <= RCMREQ and risc_gnt and not RWREN; 
HS_DONE <=NOThmreq; 

HOST.DONE <= not (LRAM_CS AND (not H_ADR(2)) AND (not H_ADR(0)) and 

hmreq); 

HWR_LMEMD AT <= (not hmreq) AND LRAM_CS AND H WR AND (not H_ADR(2)) AND 
(not H_ADR(1)) AND (not H_ADR(0)) ; 

HWR_LMEM_ADR <= (not hmreq) AND LRAMCS AND H_WR AND (not H_ADR(2)) AND 
H_ADR(1) AND (not H_ADR(0)) ; 

HRD_LMEM_DAT_L <= LRAM_CS AND H_RD AND (not H_ADR(2)) AND (not H_ADR(1)) ; 

HRD_LMEM_DAT_H <= LRAM_CS AND H_RD AND (not H_ADR(2)) AND (not H_ADR(1)) 
AND (not H_ADR(0)); 

HRD_LMEM_ADR_L <= LRAM_CS AND H_RD AND H_ADR(1); 

HRD_LMEM_ADR_H <= LRAM_CS AND H_RD AND H_ADR(1) AND (not 

H_ADR(0)); 

HRD_LMEM_DAT <= LRAM_CS AND H_RD AND (not H_ADR(2)) AND (not H_ADR(1)) AND 
(notH_ADR(0)); 

res_17 <= addum((h_lmem_adrh & hjmem_adrl) ., 'O' & b u 10"); 
end block control; 



risc_mem_out : block 
begin 

RMEM_DO_L{7 downto 0) <= 

rjmemjeg when risc_gnfc=T and WORD=T and MEM8=T and cycte_2=:T else 
LMEM_DOUT(15 downto 8) when risc_gnt=T and WORD= , 0 I and MEM8='0' and 
RMEM_ADR(0)=T else 

LMEMJX)UT(7 downto 0); 

RMEMJX)_H(15 downto 8) <= 

LMEM_DOUT(7 downto 0) when risc_gnt=T and WORD=T and MEM8=T else 
LMEM_DOUT(15 downto 8); 

end block risc_mem_out; 



local_memjn : block 
begin 

LMEMJMN_L(7 downto 0) <= 

hjmem_reg(7 downto 0) when host_gnt=T and cycle_l=T else 
hjmem_reg(15 downto 8) when host_gnfc=T and cycle_2^:T else 
RMEM_DIN(15 downto 8) when riscjnfc=T and cycle_2=T else 
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RMEM_DIN(7 downto 0); 
LMEM_DIN^H(15 downto 8) <= 

h_lmemjreg(15 downto 8) when host_gnt=T and cycle_l=T and MEM8='0' else 
RMEMJDIN(7 downto 0) when rise _gnt=T and WORD='0' and MEM8=0' and 
RMEM_ADR(0)=T else 

RMEMJ)IN(15 downto 8); 

end block local_mem_in; 



latch JmemjriscjregO : process 
begin 

wait until (CLK = T and CLK'event); 
iflaLr_datO = Tthen 

rjmem jreg <= LMEM_DOUT(7 downto 0); 
end if; 



end process latchjmeinjisc_reg0; 



local_mem_address : block 
begin 

LMEM_ADR(14 downto 1) <= 
hjmem_adih(14 downto 8) & h_lmem_adrl(7 downto 1) when host_£nt=T else 
RMEM_ADR(14 downto 1); 

end block local_mem_address; 



host_req : process 

- variable nx_state : vlbit_vector(l downto 0); 
begin 

wait until ((CLK -Tand CLK'event) or RST = 'V); 
ifRST = Tthen 

hmreq <= '0'; 

hwren <= '0'; 

wr_en_inc_adr <= X)'; 

rd_en_inc_adr <= '0'; 

hr_state <=hr_stOv; 
else 

case integer(hr_state) is 
when hr_st0 => 

if HWRJLMEM_ADR=T then 
hrjstate <=hr_stlv; 

elsif HRD_LMEM_D AT =T then 
hr_state <= hr_stlv; 
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rd_en_Jnc_adr <= T; 
clsif HWR_LMEM_DAT =T then 
hr_state <= hr_st2v; 
wr_enjnc_adr <= T; 
end if; 
when hr_stl => 

if HWR_LMEM__ADR= , 0' and HRD_LMEM_DAT =W then 
hmreq <= 1'; 
hr_state <= hr_st3v; 

end if, 
when hr_st2 => 

if HWRJLMEM _DAT ='0* then 
hmreq <= T; 
hwren <= 1*; 
hr_state <= hr_st3v; 

end if; 
whenhr_st3=> 

if host_gnt=sT and done=T then 
hmreq <= '0'; 
hwren <= '0'; 
wr_en_inc_adr <= '0'; 
rd_en_inc_adr <= '0'; 
hastate <= hr_stOv; 

end if; 
when others => 
hr_state <= b M XX rt ; 
end case; 
endif; 

end process host_req; 



rd Jiost_reg : block 
begin 

H_INBUS(15 downto 8) <= 

hjmem_reg(15 downto 8) when HRD_LMEMJDAT_H=T else 
hjmeinjidrh when HRDJ^MEM^ADRJEI=T else ~ 
W2Z22ZZ2Z*\ 
H JNJBUS(7 downto 0) <= 

hjmem_reg(7 downto 0) when HRD_LMEMJDATJL=T else 
hjmem_adrl when HRD_LMEM_ADR_L= t l~else 
B ,, ZZZZZZ2Z"; 
end block rd_host_reg; 



host jregjnisjn : block 
begin 

hjmem_adr_in <= 

res_17(15 downto 0) when incji_adr = T else 
H_OUT_BUS(15 downto 0); 

hjmem_reg_l_in <= 

LMEMJDOUT(7 downto 0) when lmem_wrhdat 1 = T else 
HJDUT_BUS(7 downto 0); 
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hjmem_reg_h_in <= 

LMEM_DOUT(15 downto 8) when lmeml6_wr_hdat_h = T else 
LMEM_DOUT(7 downto 0) when lmem8_wrjidat_h = T else 
HJDUT_BUS(15 downto 8); 

- inc Ji_adr.dk <= inc _h_adr and (not elk) and not HWR_LMEM_ADR; 
inc_h_adr_clkh <= inc _h_adr and (not elk); 
incji_adr_clkl <= incji_adr and (not elk); 
taem_wrjidat_l <= host_gnt and cycle_l and latch_data; 
lmem_wr_hdat_l_clk <= host_gnt and cycle J and latch_data and (not elk) ; 
imem8_wrjidat_h <= host_gnt and cycle_2 and latch_data and MEM8; 
lmem8_wrjidat Juclk <= host_gnt and cycled and lateh_data and MEM8 AND (not elk); 
lmeml6_wr_hdat_h <= hosLgnt and cycle_l and latch_data and (not MEM8); 
]meml6_wrjidat Ji_clk <= host_gnt and cycle_l and latch^data and (not MEM8) AND (not elk); 
aid block host_regJbus_in; 



wrJiost_adih : process 
begin 

wait until ((mcJi_adr_clkh=T and incjh_adr_clkh'event) or (HW JJvlEM_ ADR= 1 *)) ; 
if HWR JLMEM_ADR = T then 

hjmem_adrh <= h_out_bus(15 downto 8); 
else 

h_lmem_adih <= res_17(15 downto 8); 

- if HWR_LMEM_ADR = T then 

- h_lmem_adih <= hjmem_adr_in(15 downto 8); 

- hjmem_adrl <= h Jmem_adr_in(7 downto 0); 

- else 

- h_lmem_adrh <= hjmem_adrjn(15 downto 8); 

- hjmem_adrl <= h Jmem_adr_in(7 downto 0); 
end if; 

end process wrJiosLadrh; 



wrJiost_adrl : process 
begin 

wait until ((incJi_adr_cIfcfc=T and inc.hjadr.clkl'event) or (HWR_LMEM_ADR=*1*)); 
if HWR_LMEM_ADR = T then 

hbnem_adrl <= h_out_bus(7 downto 0); 
else 

h_lmem_adrl <= res_17(7 downto 0); 
end if; 

end process wr_host_adrl; 

wrjiostjregj : process(lmem_wr_hdat_LcIk, HWR_LMEM_DAT, h JmemjregJJn) 
begin 

if (lroem_wrjidatj_cte= , l *) or (HWR_LMEM_DAT = T) then 

h_lmemjeg(7 downto 0) <= hjmem_jegj_in; 
endif; 

end process wrjtostj«gj; 

wr_hostjreg_h : process (lmeml6_wrjidatjh_clk, lmem8_wrjidat Ji_clk, 
HWR_LMEM_D AT, h Jmem jregjijui) 

begin 

if (lmeml6_wr - .hdat_h_clfc= , l , or lmem8_wrjidat_h_clk =T or HWRJLMEMJDAT = T) then 
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hjmem_reg(15 downto 8) <= hjmemjregji in; 
end if; 

end process wr_Jiost_iegJi; 



local_mem_process : process 
begin 



(start_proc =T and start^proc'event ) or (xsLproc = T) or (RST=*1*V 
if RST=s T then 



lmemjjroc <= '0'; 
elsif rsrproc = T then 

Imem_proc <= '0'; 
else 

lmem_proc <= T; 
end if; 

and process local_mem_process; 



local_mem_wr_en : process 
begin 



wait until ((CLK = '0' and CUCevent) or RST d-n ■• 
ifRST=Tthen 



wr_en <= '0'; 
else 

if Imemjroc = T then 

wr_en <= T ; 
elsif latch_data ='1' then 
wr_en <= '0'; 
end if; 
end if; 

end process local_mem_wr_en; 



local_m«n_process_control : process 
begin 

wait until ((CLK a Tand CLK'event) or RST = TV 
ifRST = T then 

rst_proc <= i 0 t ; 

latch_data <= W; 

lp_state <= lp_stOv, 
else 

case integer(lp_state) is 

when lp_st0 => 

if lmem_proc==T then 

if MEMJ/fAITiiyzW and MEMJWAITflKO* then 
rst_proc <= T; 
lp_state <= lp_st4v; 
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else 

lp_state <= lp_stlv; 
aid if; 
end if; 

when lp_stl => 

ifMEM_WAIT(l)=W and MEM_WAIT(0>= , r then 

rst_proc <=T; 

lp_state <= lp_st4v; 
else 

lp_state <= lp_st2v; 
end if; 

when lp_st2 => 

if MEM_WATT(1)=T and MEMJVArr(0K0* then 

rsLproc <=T; 

lp_state <= lp_st4v; 
else 

lp_state<=Ip_st3v; 
end if; 

- when lp_st3 => 

- if MEM_WAIT(1)=T and MEM_W ATIXO)^ 1 • then 

- rst_proc <=T; 

- lp_state <= lp_st5v; 

- else 

- lp_state <= lp_st4v; 

- endif; 

when lp_st3 => 
isrproc <=T; 
lp_state <= lp_st4v; 

when lp_st4 => 

isrproc <= , 0 f ; 
latch_data<=T; 
lp_state <= lp_st5v; 

when lp_st5 => 
latch_data <= '0'; 
lp_state<=lp_stOv; 
when others => 
Ipjstate <= b"XXX n ; 
end case; 
endif; 

end process localjtnem_process_control; 



local_mem_control : process 
begin 

wait until ((CLK = T and CLK'event) or RST = TV 
ifRST = Tthen 
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hosLgnt^'O'; 

risc_gnt <= '0'; 
cycle_l <=*()'; 
cycle_2 <='0'; 
start_proc <= X)'; 
done <= '0'; 
lmc_state <=idlev; 
else 

case mteger(lmc_state) is 
when idle => 

ifHMREQ=Tthen 

start_proc <= T; 

host_gnt <= T; 

lmc_state <= cy_10v; 

cycle_l <=T; 
elsif RCMREQ=T and HMREQ= t 0 t then 

start_proc<= T; 

risc_gnt <= T; 

lmc_siate <= cy_10v; 

cycle_l <=T; 
end if; 

when cy_10 => 

starrpiioc <= '0 1 ; 
lmc_state <= cy_llv; 

whency_ll => 

if lraem^proc = *0' then 
if (risc_gnt=T and ((MEM8= , 0 t and WORD^l 1 ) or WORD='0 T )) or 
(ht>st_gnt=T and MEM8='0' ) then 
done<=T; 
lmcjstate <=lastv; 

else 

lmc_state <=cy_12v; 
endif; 
end if; 

when cy^l2 => 

start_proc <= T; 
cycle_l <=*0'; 
cycle_2 <=T; 

hnc_state <=cy_20v; 

when cy_20 => 

start_proc <= '()'; 
lmc_state <= cy_21v; 

when cy_21 => 

if Imemjroc = '0' then 

done <= T; 

lmc_state <= lastv; 
endif, 



when last => 
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done<= , 0 1 ; 
host_gnt <= W; 
rise _gnt <= "0 T ; 
cyclej <= , 0 t ; 
cycled <='0'; 
Interstate <= idlev; 
when others => 

lmc_state <= b"XXX"; 
end case; 

-host wr data : write lmem_data then inc. host lmem_adr 
-host rd data : inc. host lmem_adr then rd lmem_data 
-host wradr : rd lmem_data didn't change host lmem_adr 

if (host_gm=T and Imc_state = lastv and wr_en_inc_adn=T) 

or (hr_state = hr_stlv and HRD_LMEM_DAT= t O' and rd_en_inc__adr=T) then 
inc_h_adr <= T; 
else 

inc Ji_adr <= t)'; 
end if; 
end if; 

end process local_mem_control; 



end BEHAVIORAL; 



WO 95/06286 



PCT/US94/09415 



- 93 - 



- Hie name : risc_st.vhd 

- Create by Don Chang 

- This is synthesize model for RISC state machine 

take out all random logic put it into decoder logic file 



— library WORK; 

" use WORKLsysjpkg^ll; 

entity risc__st is 
port( 

— RISC block interface 

RCMREQroutvlbit; 
ROdDONE: in vlbit; 
WREN rout vlbit; 

- register and decoder interface 

sc_wait : in vlbit; 
halt : in vlbit; 
fast_ack : in vlbit; 

dec_2_m_r: in vlbit; 
dec_2 jm_w : in vlbit; 
dec_2_wt : in vlbit; 
dec_2_j_m : in vlbit; 
exe_2_wait : in vlbit; 
wt_2_m_w : in vlbit; 
wt_2J_m : in vlbit; 
wt_2_exec : in vlbit; 
m_\v_2_exec : in vlbit; 
m_r_2__wt : in vlbit; 
m_r_2J_m : in vlbit; 

st J routvlbir, 
stjdecode : out vlbit; 
st_execute : out vlbit; 
st^wait rout vlbit; 
st jm_r : out vlbit; 
st_j_m rout vlbit; 
sLm_w : out vlbit; 
stjdleb rout vlbit; 

RST: in vlbit; 
CLK: in vlbit 

); 

end risc_st; 

architecture BEHAVIORAL of risers t is 
signal st_fetch r vlbit; 
signal st_p_dec : vlbit; 
signal st_dec : vlbit; 
signal st_exec : vlbit; 
signal st_wt: vlbit; 



- level reset by RCMDONE 

~ pulse, local memory to RISC acknowlodge 

- 1 = write, 0 = read 
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signal st_mem_wr : vlbit; 
signal st_mem_rd : vlbit; 
signal st j m eye : vlbit; 
signal req_cDc: vlbit; 
signal sync_sc_w : vlbit; 
signal st__extn_wait : vlbit; 
signal exe__ack_wt : vlbit; 
signal mem jr_ack_wt : vlbit; 
constant idle : integer := 0; 
constant fetch : integer := 128; 
constant pre_decode : integer := 64; 
constant decode : integer := 32; 
constant execute : integer := 16; 
constant risc__wait : integer := 8; 
constant mem__wr : integer := 4; 
constant mem__rd : integer := 2; 
constant jp__mem_cyc : integer := 1; 
constant extia_wait : integer := 256; 
signal state : vlbit_vector (8 downto 0); 
begin 

misc Jogic : block 
begin 

state <= st_extia_wait & st Jfetch & st_p_dec & st__dec & 

st_exec & st_wt & stjmem__wr & st__mem_rd & st j m eye; 
WREN <= st_mem_wr; 

RCMREQ <= (stj__m__cyc or st_mem_wr or st_mem_rd or stjetch) and NOT RCMDONE; 

st Jldi <= st_j_m_cyc; 

st_m__r <= sMnem__id; 

st_m__w <= st__mem_jvr; 

st_f <= stjfetch; 

st_decode <= st__dec; 

st_execute <= st_exec; 

st_wait <= st_wt; 

sUdleb <= NOT (NOT st__fetch and NOT st_p_dec and NOT st__dec and 
NOT st_exec and NOT st__wt and NOT st_mem wr and NOT st_mem__rd and 
NOTstJ_m_cyc); 

exe_ack_wt <= NOT £ast_ack and exe_2_wait; 

mem__r_ack__wt <= NOT £ast_ack and m_.r_2._wt; 
end block misc Jogic; 

risc__st__machine : process 
begin 

wait until (RST = T)or (CLK = T and CLKEVENT); 
ifRST=Tthen 

st_extra_wait <= '0'; 

sLfetch <= '0'; 

st_mem__id <= '0'; 

stj__m_cyc <= W; 

st__mem__wr <= TP; 

st__exec <= '0'; 

st_p_dec <= '0'; 

st_dec <= '0'; 

st_wt<='0'; 
else 
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sync_sc_w <= sc_wait; 
case vld2int(state) is 
when idle => 
if halt s'O' then 
st_fetch<=T; 
end if; 



when fetch => 
ifRCMDONE = Tthen 
st_p_dec <=T; 
stjfetch <= 
endif; 



when pre.decode => 
st_j)_dec <= '0'; 
st_dec <- T; 



when decode => 
if dec_2_m_r = T then 

st_mem_rd <= 'l f ; 
elsif decJ2_m_w = T then 

st_mem_wr <=T; 
elsif dec_2_wt = T then 

st_wt <= T; 
elsif dec_2 J_m - T then 

st J _m_cyc <= T; 
else 

st_exec <= T; 
end if; 

st_dec <= *&; 



when execute => 
if exe_ack_wt = T then 

st_extra_wait <= T; 
elsif exeJ2_wait = T then 

st_wt <= T; 
elsif halt ='0' then 

stJetch<=T; 
end if; 

st_exec <= '0'; 



when risc_wait => 
if sync_sc_w = X)' then 
if wt_2_m_w = T then 

st_raem_wr <= 1'; 
elsif wt_2 j m = T then 

stj..m eye <= T; 
elsif wt_2_exec = T then 
st_exec <= T; 
elsif halt = '0' then 
st_fetch <= T; 
endif; 
st_wt <= '0'; 
endif; 
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when mem_wr=> 
if RCMDONE = T then 
if m_w_2_exec = T then 
st_exec <= T; 
elsif halt = '0' then 
stjetch <= T; 
end if; 

st_mem_wr <= 'O*; 
end if; 



when mem_rd => 
if RCMDONE = T then 
if mem_r_ack_wt = T then 

st_extra_wait<=T; 
elsif m_r2_wt = T then 

st_wt <= f f; 
elsif m_r^2 j m = T then 

stjjn_cyc<=T; 
elsif halt ^'O* then 

stjetch <= T; 
end if; 

st_mem_rd <= '0'; 
end if; 



when jp_mem_cyc => 
if RCMDONE = T then 
if halt ='()• then 
st_fetch <= T; 
end if; 

st j m eye <= '0'; * 
aid if; 

when extra_wait => 
st_extaL_wait <=•()'; 
st_wt<=T; 
when others => 
end case; 
end if, 

end process risc_st_machine; 
end BEHAVIORAL; 
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- This is synthesize model of RISC register block 

- Program histroy 

- 04/07/93 Created by Don Chang 
04/15/93 add return and stack logic 

- 04/15/93 add word to port 

- 04/22/93 take all random logic out put them into decoder logic 



libiaryWORK; 

use WORK.sys_pkg.all; 

entity risc_reg is 
port( 

count_out_bus : in vIbit_vector(15 downto 0); 
scsijn J>us : out vlbit_vector(7 downto 0); 
scsi_out_bus : in vlbit_yector(7 downto 0); 
mem_inj>us : out vlbit_vector(15 downto 0); 
mem_out_bus : in vlbit_vector(15 downto 0); 
mem_addr Jros : out v!bit_vector(14 downto 0); 
host_inJ>us : out vIbit_vector(15 downto 0); 
host_out_bus : in vIbit_vector(15 downto 0); 
r_acc_reg : out vIbit_vector (7 downto 0); 
r_insth_reg : out vIbit_vector (15 downto 6); 
r_insd_reg : out vlbit_vector (2 downto 0); 
r_alu_out : out vlbit_vector (7 downto 0); 

ph_reg_i : out vIbit_vector(2 downto 0); 
idjregj : out vlbit_vector(2 downto 0); 
id_reg_o : in vlbit_vector(2 downto 0); 

alu__and : in vlbit; 
alu_or : in vlbit; 
- alu_comp : in vlbit; 
alu_add : in vlbit; 
ix_2_alu : in vlbit; 
id_2_alu : in vlbit; 
sc_2_alu: in vlbit; 
pc_2_alu: in vlbit; 
alu_r_jump : in vlbit; 
alu_comp_i : in vlbit; 
alu_minus_l : in vlbit; 
alu _plus_2 : in vlbit; 
alu _plus_l : in vlbit; 
en_stac_ad : in vlbit; 
ahi_2_reg : in vlbit; 
inst_mvbi : in vlbit; 
alu_2 jpc : in vlbit; 
reg_2_id : in vlbit; 
mvrr_ix : in Ylbit; 

inst_ms_sel : in vlbit; 
inst jns_dma : in vlbit; 
inst_ms_ret : in vlbit; 
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inst_ms_sint : in vlbit; 
instjnsjialt : in vlbit; 

risc_idle : in vlbit; 

- setjbalt : out vlbit; 

hosrint : out vlbit; 
rstjnt : in vlbit; 

en_dma : out vlbit; 
sel_str rout vlbit; 

- decoder and register block to state machine interface signal 

stjfetch : in vlbit; 

- st_exec : in vlbit; 
stjwait : in vlbit; 



wr_pc : in vlbit; 
wrjacc : in vlbit; 
wr_qp : in vlbit; 
wr_ix : in vlbit; 
wr_ih : in vlbit; 
en_pc_d : in vlbit; 
en_ix_d : in vlbit; 
en_qp_d : in vlbit; 
enJiost_put : in vlbit; 
en_cuntr_do : in vlbit; 
en_tm_2_rj : in vlbit; 
en_sc_2_r_i : in vlbit; 
enjd_2_r_i: in vlbit; 
en_sc_2_mem : in vlbit; 
en_id_2 jmtem : in vlbit; 
en_inst_d : in vlbit; 

en_qp_ad : in vlbit; 
enJLad : in vlbit; 
en_ixq_ad : in vlbit; 
en_ixl_ad : in vlbit; 
tmout Jump : in vlbit; 



• write program counter pulse 

• write accumulator pulse 

- write Q pointer pulse 

- write index register pulse 

- write instruction holding register pulse 

• enable program counter to data bus 

- enable index register to data bus 



— enable instruction holding register to data bus 



md2scsi : in vlbit; 
reg2scsi : in vlbit; 
stac_clk : in vlbit; 
int_clk : in vlbit; 
ix_clk: in vlbit; 



); 



RST: in vlbit 



end risc_reg; 



architecture BEHAVIORAL of risc_reg is 



WO 95/06286 



PCT/US94/09415 



- 99 - 



signal pc_reg : vIbit_vector(l 1 downto 1); 
signal acc_reg : vlbitjvector(15 downto 0); 
signal qpjreg : vlbit_vector(7 downto 0); 
signal ixjreg : vlbit_vector(5 downto 0); 
signal instjttg : vlbit_vector(15 downto 0); 
signal stac jreg : vlbit; 
signal alu_out : vIbit_vector(ll downto 0); 
signal regjnjras : vlbit_vector(15 downto 0); 
signal reg^ouLbus : vlbit_vector(15 downto 0); 
signal pcJLbus : vlbit_vector(ll downto 1); 
signal inst_i_bus : vlbit_vector(15 downto 0); 
signal id_reg_x : vlbit_vector(7 downto 1); 
signal id_reg : vlbit_vector (7 downto 0); 
signal ix Jn _bus : vIbit_vector (5 downto 0); 

begin 



datajmsjogic : block 
begin 

- i<Lreg is 3 bits come from device id register 

— scsi came from scsi out bus 

- move register to register acturly move data from external register 

— to internal register 
ieg_inj>us<= 

host_out_bus(15 downto 0) after 2 ns when riscjdle = T else 
"0000" & alu_out after 2 ns when alu_2_reg = T else 
"00000000" & id_jeg after 2 ns when en_id_2_r_i = T else 
"00000000" & scsi_out_bus after 2 ns when en_sc_2_r_i = T else 

- "00000000" & tm_reg after 2 ns when enjm_2_r_i = 1' else 
instjreg after 2 ns when en_inst_d = T else 
"0000000000" & ix_reg(5 downto 0) when mvrrjx = T else 
mem_out_bus after 2 ns; 

— id_reg is 3 bits come from device id register 

- counter and pointer register are located outside 

— scsi also located outside 
reg_out_bus <= 

"0000" & pc_reg(ll downto 1) & '0* after 2 ns when en_pc_d = T else 
"00000000" & qp_reg(7 downto 0) after 2 ns when en_qp_d = T else 
"0000000000" & ix__reg(5 downto 0) after 2 ns when en_jx_d = T else 
inst_reg(15 downto 0) after 2 ns when enjnst d = T eise 
acc_ieg after 2 ns; 



ixjnj>us <= host_out_bus(13 downto 8) when riscjdle = T else 
inst_reg(5 downto 0) when inst_mvbi = T else 
acc_reg(5 downto 0) when nmrjx = T else 
mem_out_bus(5 downto 0) after 2 ns; 

scsUn_bus <= 

mem_ouLbus(7 downto 0) after 2 ns when md2scsi = T else 
reg_out_bus(7 downto 0) after 2 ns when reg2scsi = T else 
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"ZZZZZZZZ" after 2 ns; 

mem_in_bus <= 
coimLouLbus (15 downto 0) when en_cuntr_do = T else 
"00000000" & scsLout_bus (7 downto 0) when en_sc_2_jnem = T else 
"00000000" & id_reg (7 downto 0) when enjd_2_mem = T else 
reg_outJ>us(15 downto 0) after 2 ns; 

host_in_bus <= 
"ZZZZZZZZZZZZZZZZ" when en _host_out = *0' else 
qp_reg(7 downto 0) & inst_reg(7 downto 0) when en_qp_d = T else 
"00" & ixjreg(5 downto 0) & acc_reg(7 downto 0) when en_ix_d = T else 
reg_out_bus; 

pc JUbos <= 

alu_out(10 downto 0) after 2 ns when ahiJLpc = T else 
regjn_bus(ll downto 1) after 2 ns; 



idjregjc <= 

reg_out_biis(7 downto 1) after 2 ns when reg_2jd = T else 

mem_out_bus(7 downto 1) after 2 ns; 
conv_id_i : block 
begin 

id_reg_i(0) <= id_reg_x(7) or id_reg_x(5) or id_reg_x(3) or id_reg_x(l); 
id_reg_i(l) <= id_reg_x(7) or id_reg_x(6) or id_reg_x(3) or id_reg_x(2); 
id_regj(2) <= id_reg_x(7) or id_ieg_x(6) or id_reg_x(5) or id_reg_x(4); 
end block convjdj; 

inst_i_bus <= 

host_outJ>us(15 downto 0) when riscjdle = T else 
mem_out_bus; 

end block data^bnsjogic; 



- sint, rflag, sel, dma 
miscjnstjogic : block 
begin 

- halt logic change into decode and latch locate in control register block 

- halt Jogic : process 

- begin 

- wait until ((st_exec = T and st_exec'event) or RST = T or clr_halt = T or setjialt s , l f ); 

- ifRSTsT then 

risc_idle<=T; 

- elsif setjialt = T then 

- risc_idle <= T; 

- ; elsif clrhalt = T then 

- rise Jdle <= '0 1 ; 

- else 

- risc_idle <= inst_ms_halt; 

- end if, 
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- end process halt Jogic; 



seljogic: process 
begin 

wait until ((st_wait = T and st_wait*event) orRST= T or st fetch = TV 
ifRST = Tthen 

seLstr <=s 'O'; 
elsifst_fetch = Tthen 

seLstr<= , 0 1 ; 
else 

sel_str <= inst ms sel; 
endif; 
aid process seljogic; 

sintjogic : process 
begin 

wait until ((int_clk = T and inLclk'event) or RST = T or rst int = TV 
ifRST = Tthen 

host.inKs'O 1 ; 
elsifrstjnt= Tthen 

hosUnKJs'O'; 



host_int <= inst_ms_sint or inst_msjialt; 
endif; 
end process sintjogic; 

en_dma <= inst_ms_dma and st_wait; 
end block miscjnst Jogic; 

addrjwsjogic : block 



mem_addrj>us <= 

W & pc_reg(ll downto 1) & V when pcJLalu = T and tmoutjump = '0* else 
OOOOOOOOUlir &(inst^ms^retxorstacjeg)& , 0 , whenen stac ad =Telse 
n 000000001111010"whentmomjmnp = l»else ~ ~ 

"00000000" & inst_reg(6 downto 0) when enj_ad = T else 
T & qp_reg(7 downto 0) & ixjreg(5 downto 0) when en_ixq_ad = T else 
"OOOOOOOOO" & ix_reg(5 downto 0) when enjxl_ad = T else 
T & qp_reg(7 downto 0) & inst_reg(5 downto 0); 
end block addrjnisjogic; 



stacjregister : process 
begin 

wait until ((stac_clk = V and stac clk'event) or RST = TV 
ifRST=Tthen ~ ■ 

stacjreg <= T; 
else 

stac_reg <= NOT stac_reg; 
endif; 
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program_counter : process 
begin 

wait until (wr_pc = '0' and wr_pc'event); 
pc_reg<=pcXbus; 
end process program_counter, 



accumulator : process 
begin 

wait until (wr_acc = '0' and wr_acc*event); 
acc jreg <= regjnjras; 
end process accumulator, 



<LPointer_reg : process 
begin 

wait until (wr_qp = '0' and wr qp'event); 
if rise Jdle='0' then 
qpjeg <= regjn _bus(7 downto 0); 



<B>_ieg <= hosLout_bus(15 downto 8); 
endif; 

end process q_pointer_reg; 



index_reg : process 
begin 

wait until ((ix_clk « 9 ff and ix elk'event) orwt ix = TV 
ifwrjx = T then " 

k_reg <= ixjn_bus; 
else 

»L.reg <= alu_out(5 downto 0); 
endif; 



instJiold_reg : process (wrjh, inst JLbus) 
begin 

- wait until (wrjh = '0' and wrjh'event); 
if wrjh = Tthen 

mst _reg <= instj_bus(15 downto 0); 
endif; 
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ahi: block 

signal alu_in_a : vlbit_vector(ll downto 0); 
signal alu_in_b : vlbit_vector(ll downto 0); 

begin 

alujn_a <= 

"000000000001" when alu_plus_l = T else 
"000000000010" when akrplus 2 = T else 
"UllllllUir when alu_minus_l = T else 
inst_reg(l 1 downto 0) when alu_compJi = T else 
inst_reg(8) & instjreg(8) & instjreg(8) & instjreg(8 downto 0) 

when alu_rjnmp = 1' else 
mem_out_bus(ll downto 0); 

alujnj) <= 

"0" & pc_reg(l 1 downto 1) after 2 ns when pc_2_alu = T else 
"0000" & scsi_out_bus after 2 ns when sc_2_alu = T else 
"0000" & id_reg after 2 ns when id_2_alu = T else 
"000000" & ixjeg after 2 ns when ix_2_alu = T else 
acc_reg(l 1 downto 0) after 2 ns; 

alu_out <= 

add_12bit (alu_in_a, alu_in_b) after 2 ns when alujadd = , r else 
"0000" & or_8bit (alu_in_a, alu_inj>) after 2 ns when alu_or = T else 
"0000" & and_8bit (alu_jn_a, ahijnj)) after 2 ns when alu_and = T else 
"0000" & comp_8bit (alujna, alu_inj>) after 2 ns; 
end block alu; 

out_signal : block 
begin 

r_accjreg <= acc_reg(7 downto 0); 
rjnsthjeg <= instjceg(15 downto 6); 
rjnstl_reg <= inst_ieg(2 downto 0); 
r_alu_out <= alu_out(7 downto 0); 
ph_reg_i <= instjreg(2 downto 0); 
- halt <= rise Jkfle; 
conv_id_o : block 
begin 

i<Lieg(0) <= NOT id_reg_o(2) and NOT id_regj)(l) and NOT id reg_o(0); 
id_ieg(l) <= NOT idjreg_o(2) and NOT id_reg_o(l) and id_reg_o(0); 
id_reg(2) <= NOT id_reg_o(2) and id_regj>(l) and NOT id_ieg_o(0); 
id_reg(3) <= NOT id_reg_o(2) and id_reg_o(l) and id_reg_o(0); 
id_reg(4) <= id_reg^o(2) and NOT id_reg_o(l) and NOT id_reg_o(0); 
id_reg(5) <= id_reg_o(2) and NOT id_reg_o(l) and id_reg_o(0); 
id_reg(6) <= id_reg_o(2) and id_reg_j>(l) and NOT id_reg_o(0); 
id_reg(7) <= id_reg_o(2) and id_reg_o(l) and id_reg_o(0); 
end block conv_id_o; 
end block out_signal; 
end BEHAVIORAL; 
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- This is synthesize model for RISC decoder logic 



— library WORK; 

- use WORK.sys_pkg.all; 

entity decode is 
port( 

CLK : in vlbit; 
rst : in vlbit; 
RCMDONE : in vlbit; 

insth : in vlbit_vector (15 downto 6); 

instl : in vIbit_vector (2 downto 0); 

alu_and : out vlbit; 

ato_pr : out vlbit; 

alu_comp : out vlbit; 

alu_add : out vlbit; 

ix_2_alu : out vlbit; 

id_2_alu : out vlbit; 

sc_2_alu : out vlbit; 

pc_2_alu : out vlbit; 

alu_rjump : out vlbit; 

alu_comp_i : out vlbit; 

alu_minus_l : out vlbit; 

alu _plus_2 : out vlbit; 

alu _plus_l : out vlbit; 

en_stac_ad : out vlbit; 

alu_2_reg : out vlbit; 

d_inst_mvbi : out vlbit; 

alu J2 _pc : out vlbit; 

reg_2 _id : out vlbit; 

mvirjx : out vlbit; 

alu_out : in v!bit_vector (7 downto 0); 

riscjdle : in vlbit; 
setjialt : out vlbit; 
rset_am : out vlbit; 
d_inst__ms_sel : out vlbit; 
d_inst_ms_dma : out vlbit; 
d_inst_ms_ret : out vlbit; 
d_inst_ms_sint : out vlbit; 
d Jnst_ms_halt : out vlbit; 

- decoder and register block to state machine interface signal 
sLfetch : in vlbit; 
st_exec : in vlbit; 
st j m : in vlbit; 
st_wait : in vlbit; 
st_m_r : in vlbit; 
st_m_w : in vlbit; 
stjiec : in vlbit; 
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wr_pc : out vlbit; 
wr_acc : out vlbit; 
wr_qp : out vlbit; 
wr_ix : out vlbit; 
wrjh : out vlbit; 
en_pc_d : out vlbit; 
en_ix_d : out vlbit; 
en_qp_d : out vlbit; 
en_cuntr_do : out vlbit; 
en_inst_d : out vlbit; 



• write program counter pulse 

- write accumulator pulse 

• write Q pointer pulse 

• write index register pulse 

- write instruction holding register pulse 

- enable program counter to data bus 

• enable index register to data bus 



- enable instruction holding register to data bus 



en_qp_ad : out vlbit; 
en J_ad : out vlbit; 
en_ixq_ad : out vlbit; 
enjxl_ad : out vlbit; 
tmoutjump : out vlbit; 
d_md2scsi : out vlbiq 
d_jeg2scsi : out vlbit; 
en_id_2__mem : out vlbit; 
en Jd__2_r J : out vlbit; 
en_sc_2_mem : out vlbit; 
en_sc_2_r J : out vlbit; 
en_host_out ; out vlbit; 

stac_cflc: out vlbit; 
int_clk : out vlbit; 
ix_clk : out vlbit; 

dec_2_m_r : out vlbit; 
dec_2_m_w : out vlbit; 
dec_2_wt : out vlbit; 
dec_2_j_m : out vlbit; 
exe_2_wait : out vlbit; 
wt_2_m_w : out vlbit; 
wt_2_j_m : out vlbit; 
wt_2_exec : out vlbit; 
m_w_2_exec : out vlbit; 
m_r_2_wt : out vlbit; 
m_r_2 j m : out vlbit; 
sc_wait : out vlbit; 



r_wr_scsi : out vlbit; 
r_wr_ph: out vlbit; 
r_wr_id : out vlbit; 

bc_xp_adr : out vIbit_vector(l downto 0); 

bc_xp_cs : out vlbit; 

bc_xp_wr : out vlbit; 

r_wr_bc01 : out vlbit; 

rjwr _bc23 : out vlbit; 

r_wr_xp01 : out vlbit; 

r_wr_xp23 : out vlbit; 

r_jd_bc01 : out vlbit; 

r_rdj>c23: out vlbit; 

r_rd_xp01 : out vlbit; 
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rjd_xp23 : out vlbit; 

rstjlg_bsyb : out vlbit; 
rst_flg_ack : out vlbit; 
rst_flg_am : out vlbit; 
rst Jlgjprty : out vlbit; 
rst_free_tm : out vlbit; 
wr_w_d_tm : out vlbit; 

sc_pib : out vlbit; 
d_word : out vlbit; 

acc_reg : in vlbit_vector (7 downto 0); 

sec_tm_out : in vlbit; 
time-out : in vlbit; 
parity_err : in vlbit; 
reelected : in vlbit; 
selected : in vlbit; 
bc_zero : in vlbit; 
seLdone : in vlbit; 
hdshk_done : in vlbit; 
dma_done : in vlbit; 
ph_ok : in vlbit; 
req_on : in vlbit; 
busjree : in vlbit; 
atnib : in vlbit; 

hostjxn : in vlbit; 
hostjhn : in vlbit; 
hosLqpn : in vlbit; 
host_accn : in vlbit; 
host^pcn : in vlbit; 

- hostjrdn : in vlbit; 
host_wr_ix : in vlbit; 
host_wrjh : in vlbit; 
host_wr_qp : in vlbit; 
host_wr_acc : in vlbit; 
host_wr_pc : in vlbit 

); 

end decode; 

architecture BEHAVIORAL of decode is 

signal instjnove : vlbit; 
signal instjump : vlbit; 
signal inst_misc : vlbit; 

- Q pointer offset addressing 

signal mv_rq : vlbit; 

- direct local memory addressing 

signal mv_rl: vlbit; 

- index addressing 

signal mv_x: vlbit; 
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— register to register move 



signal 


mv_rr : vlbit; 


— immediate data move 


signal 


mv_i ; vlbit; 


signal 


mv_wi : vlbit; 


signal 


mv_bi : vlbit; 


signal 


inst_jnvbi : vlbit; 


signal 


inst_mvrq : vlbit; 


signal 


inst_mvrl : vlbit; 


signal 


inst_mvx : vlbit; 


signal 


inst_mvrr : vlbit; 


signal 


inst_mvwi : vlbit; 


-flag test jump 




signal 


jp_tstf : vlbit; 


-bit test jump 




signal 


jp_tstb : vlbit; 


-compare jump 




signal 


jp_comp : vlbit; 


- long jump and call 


signal 


jp_call : vlbit; 



signal inst Jp_t_f : vlbit; 

signal instjp_tj> : vlbit; 

signal instjp_c_q : vlbit; 

signal inst_jp__c_i : vlbit; 

signal inst Jpx : vlbit; 

signal inst_call : vlbit; 
-jump condition 

signal truejump : vlbit; 

signal jump_true : vlbit; 

signal chkjsel : vlbit_vector (2 downto 0); 

- misc instruction 

signal ms_and : vlbit; 

signal ms_or : vlbit; 

signal ms_xor : vlbit; 

signal msjncr : vlbit; 

signal ms_decr : vlbit; 

- register input bus control 

signal ms_reg_op : vlbit; 

signal inst_ms_orq : vlbit; 

signal inst_ms_orl : vlbit; 

signal inst jms.andq : vlbit; 

signal inst_ms_andl : vlbit; 

signal inst_ms_xorq : vlbit; 

signal inst_ms_xorl : vlbit; 

signal inst_ms_incr : vlbit; 

signal inst_ms_decr : vlbit; 

- the rest miscellaneous instruction decode 

signal inst_ms_rflag : vlbit; 
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signal 


inst_ms_ret : vlbit; 


digital 


4ncf m c cinf • vlHJt* 
mal_IIlo__buH . VIUll, 


signal 


inst_ms._dma : vlbit; 


signal 


inst_ms_w_f : vlbit; 


signal 


inst_ms_sel : vlbit; 


signal 


inst_ms_halt : vlbit; 


signal 


inst_word : vlbit; 


signal 


word : vlbit; 


signal 


reg_out : vlbit; 


- register address 


signal 


reg_acc_adr : vlbit; 


signal 


reg_qp_adr : vlbit; 


signal 


reg_ph__adr : vlbit; 


signal 


reg_ix_adr : vlbit; 


signal 


reg_id_adr : vlbit; 


signal 


reg__pc_adr : vlbit; 


signal 


reg_sc_adr : vlbit; 


- signal 


reg_xp01_adr : vlbit; 


- signal 


reg_xp23_adr : vlbit; 


- signal 


reg_bc01_adr : vlbit; 


- signal 


reg_bc23_adr : vlbit; 


- exception register address 


signal 


ieg_accxi_adr : vlbit; 


signal 


reg_scx_adr : vlbit; 


signal 


reg_idx__adr : vlbit; 


signal 


reg_ixx_adr : vlbit; 


- flag address 




signal 


flag_bsy : vlbit; 


signal 


flag_ack : vlbit; 


signal 


flag_am : vlbit; 


signal 


fla£_set_atn : vlbit; 


signal 


flag prty : vlbit; 


signal 


flagjtm : vlbit; 


signal 


flag_wdtm : vlbit; 


signal 


en_ix_ad : vlbit; 


signal 


md2scsi : vlbit; 


signal 


rcg2scsi : vlbit; 


— alu operation 




signal 


alu_plus_l_x : vlbit; 



signal alu _plus_J2_x : vlbit; 

signal alu_mimis_l_x : vlbit; 

signal alu_r Jomp_x : vlbit; 

- watch dog timer 

signal wr_wd_tm : vlbit; 

signal time_out_set : vlbit; 

signal time_out_clk : vlbit; 
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signal tmouijumpj : vlbit; 

begin 

inst_decJogic : block 
begin 

— instruction class 

inst_move <= NOT insth(15) and NOT insth(14); 

inst Jump <= insth( 1 5); 

inst_misc <= NOT insth(15) and insth(14); 

— Q pointer offset addressing 

mv_rq <= NOT insth(9) and insth(8); 

— direct local memory addressing 

mv_rl <= NOT insth(9) and NOT insth(8); 

— index addressing 

mv_x <= insth(9) and NOT insth(8); 

— register to register move 

mv_rr <= NOT insth(13) and insth(9) and insth(8); 

— immediate data move 

mv_i <= insth(13) and insth(9) and insth(8); 
mv_wi <= mv_i and insth(12); 
mv J>i <= mvj and NOT insth(12); 

inst_mvbi <= inst_move and mv Jbi; 
mst_mvrq <= inst_move and mvjq; 
inst_mvrl <= insMnove and mv_rl; 
inst_mvx <= inst_move and mv_x; 
inst_mvrr <= inst__move and mv_rr, 
inst_mvwi <= inst_move and mv_wi; 

— flag test jump 

jp_tstf <= insth(14) and insth(13); 

— bit test jump 

jp_tstb <= insth(14) and NOT insth(13); 

— compare jump 

jp_comp <= NOT insth(14) and insth(13); 

— long jump and call 

jp_call <= NOT insth(14) and NOT insth(13); 

inst Jp jj <= inst Jump and jpjstf ; 
instjp_tj> <= inst Jump and jp_tstb; 
inst Jp_c_q <= inst Jump and insth(8) and jp_comp; 
inst Jp_c J <= inst Jump and NOT insth(8) and jp_comp; 
inst Jpx <= inst Jump and jp_call and NOT insth(12); 
inst_call <= inst Jump and jp_call and insth(12); 
-jump condition 
true Jump <= insth(9); 
chk_sel <= insth(12 downto 10); 

-misc instruction 
ms_and <= insth(13) and NOT insth(9); 
ms_or <= NOT insth(13) and NOT insth(9) and NOT insth(7); 
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ms_xor <= NOT insth(13) and NOT insth(9) and inslh(7); 
msjncr <= NOT insth(13) and insth(9) and insth(8) and insth(7); 
ms_decr <= NOT insth(13) and insth(9) and insth(8) and NOT insth(7); 

- register input bus control 

ms_reg_op <= inst_misc and (ms_and or ms_or or ms_incr or ms_decr); 

inst_ms_orq <= inst_misc and ms_or and insth(8); 
inst_ms_orl <= inst_misc and ms_or and NOT insth(8); 
inst_ms_xorq <= inst_misc and ms_xor and insth(8): 
inst_ms_xorl <= inst_misc and ms_xor and NOT insth(8); 
inst_ms_andq <= inst_misc and ms_and and insth(8); 
inst_ms_andl <= instjmisc and ms_and and NOT insth(8); 
inst_ms_incr <= inst_raisc and ms_incr, 
inst_ms_decr <= inst_misc and ms_decr, 

- the rest miscellaneous instruction decode 

inst_ms_ret <= inst_misc and insth(13) and insth(9) and insth(8) and NOT insth(7); 
inst_ms_sel <= inst_misc and NOT insth(13) and NOT insth(lO) and insth(9) and 

NOT insth(8) and NOT insth(7); 
inst_ms_wj: <= inst_misc and NOT insth(13) and insth(lO) and insth(9) and 

NOT insth(8) and NOT insth(7); 
inst_ms_dma <= instjnisc and NOT insth(13) and insth(9) and NOT insth(8) and insth(7); 
inst_msjrflag <= inst_misc and insth(13) and insth(9) and insth(8) and insth(7); 
inst_ms_sint <= inst_misc and insth(13) and insth(9) and NOT insth(8) and NOT insth(7); 
inst_msjialt <= inst_misc and insth(13) and insth(9) and NOT insth(8) and insth(7); 

inst_word <= inst.move and insth( 13) and NOT mvj>i; 

word <= (inst_move and insth(13) and (NOT mvj)i or NOT regjd_adr)) or 

(inst Jump and NOT(jp_comp and stjnjr)) or 

inst_ms_ret or st_fetch; 

redout <= insth(7); 

- register address 

reg.acc_.adr <= NOT insth(12) and NOT insth(ll) and NOT insth(10); 
reg_qp__adr <= NOT insth(12) and NOT insth(l 1) and insth(lO); 
reg_ph_adr <= NOT insth(12) and NOT insth( 1 1) and insth(10); 
reg_jx_adr <= NOT insth(12) and insth(l 1) and NOT insth(10); 
reg_id_adr <= NOT insth(12) and insth(ll) and NOT insth(10); 
reg_pc_adr <= NOT insth(12) and insth(l 1) and insth(10); 
reg_sc_adr <= NOT insth(12) and insth(l 1) and insth(lO); 

- exception register address 

reg_accxi_adr <= insth(12) and NOT insth(ll) and NOT insth(10); 
reg_scx_adr <= NOT instl(2) and instl(l) and instl(0); 
regjdx.adr <= NOT instl(2) and instl(l) and NOT instl(O); 
reg_ixx_adr NOT instl(2) and NOT instl(l) and instl(O); 

- flag address 

flag_ack <= NOT insth(12) and NOT insth(l 1) and NOT insth(lO); 
flag_atn <= NOT insth(12) and NOT insth(l 1) and insth(10); 
flagjnty <= NOT insth(12) and insth(l 1) and NOT insth(10); 
flag_ftm <= NOT insth(12) and insth(l 1) and insth(lO); 
flag_wdtm <= insth(12) and NOT insth(l 1) and NOT insth(lO); 
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flag.bsy <= insth(12) and NOT insih(ll) and insth(lO); 
flag.set_atn <= insth(12) and insth(l 1) and NOT insth(10); 

- counter and pointer address 
bc_xp_adr<= insth(ll downto 10); 

- reg.xp01.adr <= insth(12) and NOT insth(l 1) and NOT insth(lO); 

- reg.xp23.adr <= insth(12) and NOT insth(l 1) and insth(10); 

- reg.bc01.adr <= insth(12) and insth(l 1) and NOT insth(lO); 

- reg.bc23.adr <= insth(12) and insth(l 1) and insth(10); 

end block inst.dec.logic; 



control.logic : block 
signal reg_wr : vlbit; 
begin 

- register output bus control 

- default acc register 

en_pc.d <= (inst.mvrq and ieg_pc_adr and inst.word and NOT risc.idle) or 
(inst.mvrl and reg.pc.adr and inst.word and NOT risc.idle} or 
(inst.mvx and reg.pc.adr and inst.word and NOT risc.idle) or 
(inst Jpx and st.m.w and NOT risc.idle)or 
(inst.call and st.m.w and NOT risc.idle) or 
(NOT host j>cn and risc.idle); 

en.qp.d <= (inst.mvrq and reg.qp.adr and NOT inst.word and NOT risc.idle) or 
(inst.mvrl and reg.qp.adr and NOT inst.word and NOT risc.idle) or 
(inst.mvx and reg.qp.adr and NOT inst.word and NOT risc.idle) or 
(inst.mvrr and reg_qp.adr and NOT inst.word and NOT risc.idle) or 
(NOT host.qpn and risc.idle); 

en.ix.d <= (inst.mvrq and reg.ix.adr and NOT inst.word and NOT risc.idle) or 
(inst.mvrl and reg_ix.adr and NOT inst.word and NOT risc.idle) or 
(inst.mvx and reg.ix.adr and NOT inst.word and NOT risc.idle) or 
(NOT host.ixn and risc.idle); 

en.inst.d <= (inst.mvbi and NOT risc.idle) or 

(inst Jpx and st.exec and NOT risc.idle) or 
(inst.call and st.exec and NOT risc.idle) or 
(NOT host.ihn and risc.idle); 

- memory data bus control 

- default register data out bus 

en.id.2.mem <= (inst.mvrq and inst.word and reg.id.adr); 
en.sc.2.mem <= (inst.mvrq and reg.sc.adr and NOT inst.word) or 

(inst.mvrl and reg.sc.adr and NOT inst.word) or 

(inst.mvx and reg.sc.adr and NOT inst.word); 

- en.cuntr.do <= (inst.mvrl or inst.mvrq or inst.mvx) and 

(reglbcOl.adr or reg.bc23.adr or reg.xp01.adr or reg.xp23.adr); 

- register in bus control 

- default memory data out bus 

alu.2_reg <= inst.ms.incr or inst.ms.decr or inst.ms.andl or inst.ms.orl or 
inst.rns.andq or inst.ms.orq or inst.ms.xorq or inst.ms.xori; 
en.id.2.r.i <= inst.mvrr and reg.idx.adr; 
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en_sc_2_rj <= inst_mvrr and reg_scx_adr; 

- scsi data bus control 

md2scsi <= (inst_mvrq and reg_sc_adr and NOT inst_word and not rise Jdle) or 
(inst_mvrl and reg_sc_adr and NOT inst_word and not riscjdle) or 
(inst_mvx and reg_sc_adr and NOT inst_word and not risc_idle); 
. reg2scsi <= (instjnvrr and reg_scx_adr and reg_out and not rise Jdle) or 
(inst__mvbi and reg_sc_adr and not risc_idle); 

- memory address bus Control 
-default en_qp_ad 

- en_pc_ad = pc_2_alu 

- en_qp_ad <= inst_mvrq or inst_ms_orq or inst_ms_andq or 

- (inst Jp_ci.q and st_m_r); 

enj_ad <= inst_mvrl or inst_ms_orl or inst_ms_andl or inst_jns_xorl; 

en_ix_ad <= inst_mvx; 

en_stac_ad <= inst_call or inst_ms_ret; 

- program counter bus control . 

- default register in bus 

alu_2_pc <= st Jetch or instjp_tj> or inst Jp_t_f or (st..j m and NOT jump jrue) or inst_mvwi; 

- id register output bus control 
reg_2_id <= insMnvrr, 

- register write control logic 

- internal register write logic 

reg_wr <= st_m_r and RCMDONE and NOT CLK; 
wr_acc <= ((inst_mvrl or inst__mvrq or inst__mvx) and 

NOT reg_out and reg_acc_adr and reg_wr) or 

(inst_mvwi and rcg_wr and reg_accxi_adr) or 

(inst_mvbi and reg_acc_adr and st_exec) or 

(inst_mvrr and NOT reg_put and reg_acc_adr and st_exec) or 

(inst_ms_jncr and reg__acc_adr and st_exec) or 

(inst_ms_decr and reg_acc_adr and st_exec) or 

((inst_ms_ori or inst_ms_andl or inst_ms_orq or inst_ms_andq or 
inst_ms_xori or instjms_xorq) and reg_acc_adr and regjwr) or 

host_wr_acc; 

wrjx <= (inst_move and NOT inst_word and NOT reg_out and reg_ix_adr and 
reg_wr and (mv_rl or mv_rq)) or 
(inst_mvbi and reg_ix_adr and st_exec) or 
(inst_mvrr and reg_out and reg_ixx_adr and st_exec) or 
host_wr_ix; 

ix_clk <= (en_ix_ad and st_m_r and RCMDONE and NOT CLK) or 

(en_ix_ad and st_m_w and RCMDONE and NOT CLK); 
wr_ih <= (st_feteh and RCMDONE and NOT CLK) or 

host_wr_ih; i ^. 

wr_qp <= ((inst_mvrl or inst_mvrq) and NOT inst_word and NOT reg_out and 
reg_qp_adr and reg_wr) or 

(inst_mvrr and NOT inst_word and NOT reg_out and reg_qp_adr and 
st_exec) or 
host_wr_qp; 
wr_pc<= 

((insunvrl or inst__mvrq) and insLword and NOT reg_put and reg_pc_adr and regjwr) or 
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(jump_true and st_exec and NOT CLK and (inst Jp_t_b or inst Jp_t_f)) or 

(stj_m and RCMDONE and NOT CLK) or - jump comp 

((inst Jpx or inst_call) and st^exec and NOT CLK) or 

(st_fetch and RCMDONE and NOT CLK) or 

(inst_mvwi and reg_wr) or 

(inst_ms_ret and reg_wr) or 

host_wr_pc; 

staq_clk <= (inst_ms_ret and reg_wr) or finst_call and st_exec); 

int_clk <= (inst_ms_sint and st_exec) or 

(inst_ms_halt and st_exec and instl(0)); 

- external register write logic 

r_wr_scsi <= (md2scsi and reg_wr) or (reg2scsi and st_exec); 
r_wr_ph <= (inst_mvbi and reg_j>h_adr and st_exec) or 

(inst _jp_c_i and regjph_adr and st_dec); 
rjvrjd <= (instjnove and inst_word and reg_id_adr and NOT reg_out and 

reg_wr) or 

(inst_mvrr and reg_idx_adr and reg_out and st_exec); 
bc_xp_cs <= instjnove and inst_word and NOT mvjri and NOT (mv_wi and reg_accxi_adr) and 

insth(12); 
bc_xp_wr <= reg_wr, 

- r_wr__bc01 <= (instjrnove and inst_word and reg__bc01_adr and reg_wr); 

- r_wr_bc23 <= (inst_move and insMvord and xeg Jx:23_adr and reg_wr); 

- r_wr_xp01 <= (inst_move and inst_word and reg_xp01_adr and reg_wr and NOT mv_wi); 

- r_wr_xp23 <= (instjnove and inst_word and reg_xp23_adr and reg_wr); 

- rjd_bc01 <= inst_word and reg_bc01_adr and reg_out and 

- (inst_mvrl or inst_mvrq or inst_mvx); 

- r_jd_bc23 <= inst_word and reg_bc23_adr and reg_out and 

- (inst_mvrl or inst_mvrq or inst_mvx); 

- r_rd_xp01 <= inst_word and reg_xp01_adr and reg_out and 

- (inst_jnvrl or inst_mvrq or inst__mvx); 

- r_rd_xp23 <= inst_word and ieg_xp23_adr and reg_out and 

- (inst_mvrl or inst_mvrq or inst_mvx); 

- reset flag these signal should send to outside 

rst_flfL.bsyb <= NOT (inst_ms_rflag and flag_bsy and st_exec); 
rst_flg_ack <= inst_ms_rflag and flag_ack and st_exec; 
rst_flg_atn <= instjns_rflag and flag_atn and stjexec; 
rset_atn <= inst_ms_rflag and flag_set_atn and st_exec; 
*st_flg_prty <= inst_ms_rflag and flagj>rty and st_exec; 
rstjreejm <= inst_ms_rflag and flag_f tm and stjexec; 
wr_wd_tm <= instjns_rflag and flag_wdtm and st_exec and NOT elk; 
set Jialt <= instjns_hait and (st_exec or st_dec) and NOT elk; 

- scsi handshake enable signal 

sc_pio <= (st_wait and (instjnove and mvjr and reg_scx_adr)) or 
(st_wait and instjnove and reg_sc_adr and 
<mvjl or mv_rq or mv_x)) or 
(st_wait and inst jnvbi and reg_sc_adr); 

-branchlogic 
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- compare Q, compare immediate, test bit, test flag 
because program counter uses same adder-comparator comp_equal has to be latched 

jump_control : block 

signal latch_equal : vlbit; 

signal chk_bit : vlbit; 

signal chk_flag : vlbit; 

signal comp Jatch : vlbit; 

signal comp_equal : vlbit; 
begin 

comp_latch <= (instjp_c_i and st_dec) or 

(inst Jp_c_q and reg_wr); 
comp_equal <= NOT alu_out(7) and NOT alu_out(6) and NOT alu_put(5) and 

NOT alu_out(4) and NOT alu_out(3) and NOT alu_out(2) and 

NOT ahrout(l) and NOT alu_out(0); 

latch_comp : process (comp Jatch, comp_equaI) 
begin 

if (compjatch = T) then 
latch_equal <= comp_equal; 
end if; 
end process latch_comp; 

with vld2int(chk_sel(2 downto 0)) select 
chk_bit <= acc_reg(7) when 7, 
acc_reg(6) when 6 t 
acc_reg(5) when 5» 
acc_reg(4) when 4, 
acc_reg(3) when 3, 
acc_reg(2) when 2, 
acc_reg(l) when 1, 
acc_reg(0) when others; 
with vld2int(chk_sel(2 downto 0)) select 
chk_£lag <= NOT amib when 6, 
sec_tm_out when 5, 
parity_err when 4, 
reselected when 3, 
selected when 2, 
bc_zero when 1, 
seLdone when others; 
jump_true <= ((latch_equal and jp_comp and NOT reg ph adrt or 
(jp_comp and reg_ph_adr and ph_ok) or 
(chkjrit and jpjstb) or (chkjlag and jp_tstf)) xor 
NOT true Jump; 
end block jump_control; 
sc_wait <= NOT ((inst Jump and req_on) or 

(inst_ms_sel and (selected or reselected or seLdone)) or 
(inst_move and hdshk_done) or 
(inst_ms_dma and dma_done) or 
(inst_ms_w_f and busjfree) or tmout Jump J); 

end block controljogic; 

afoLCOntrol : block 
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begin 

alu_and <= inst_ms_andl or inst_ms_andq; 
alu_or <= inst_ms_prl or inst_ms_orq; 
- alu_comp <= instjp_c_q or inst Jp_c_i; 
alu_add <= alu_plus_2_x or alu_plus_l_x or alu_minus_l_x or alu_rjump_x; 
ix_2_alu <= inst_mvx; 
\ id_2_alu <= inst Jp_c_q and reg_id_adr, 

sc_2_ah <= (instjp_c_q or instjp_c_i) and reg_sc_adr t 

pc_2_alu <= stjfeteh or st j m or instjp>^t_f or inst JpjtJ> or inst_mvwi; 

alu_rjump_x <= inst jp_t_f or inst Jp_(_b; 

alu_comp_i <= instjp_c_i; 

alu_minus_l_x <= inst_ms_decr; 

alu_plus_2_x <= inst_mvx and insMvord; 

alu_plus_J_x <= (inst_mvx and NOT inst_word) or inst_msjncr or 

st_,fetch or st_j m or inst mvwi; 
end block aiu_control; 

state_decode : block 
begin 
dec_2_m_r <= 

inst Jp_c_q or insMnvwi or inst_ms_ret or inst_ms_andq or inst_ms_andl or 

inst_ms_orq or inst_ms_orl or inst_ms_xorq or inst_ms_xorl or 
(NOT reg_out and (inst_mvrl or inst_mvrq or inst_mvx)); 
dec_2_m_w <= 

(reg_out and NOT reg_sc_adr and NOT word and 

(inst_mvrl or inst_mvrq or inst_mvx)) or 
(reg_out and word and 

(inst_mvrl or instjmvrq or inst__mvx)) or 
inst_call; 

dec_2_wt <= (inst_mvir and reg_scx_adr and not reg_out) or 

(NOT word and reg_sc_adr and reg_out and 
(inst_mvrl or inst_mvrq or inst_jnvx)) or 

(instjp_c_i and reg_ph_adr) or 

inst_ms_sel or inst_ms_dma or inst_ms_w_f; 
dec_2J_m <= instjp_c_i and NOT reg ph adn - jump compare I 

exe_2_wait <= (inst_mvbi and reg_sc_adr) or 

(instjmvrr and reg_scx_adr and reg_out); 
wt_2jm_w <= (reg_sc_adr and reg_out and 

(inst_mvrl or inst_mvrq or inst_mvx)); 
wt_2J_m <= inst Jp_c_i and reg ph adn 
wt_2_exec <= inst_mvrr and not reg_out and reg_scx_adr, 
m_w_2_exec <= inst_call; 

m_r_2_wt <= NOT reg_out and reg_sc_adr and NOT word and 
(inst_jnvrl or inst_mvrq or inst_mvx); 
• m r 2 j m <= inst jp_c_q; 

time_out_set <= st_wait and time_out; 
time_out_clk <= tmout Jump _i and stjfeteh; 
end block state_decode; 

latchjimeout : process 
begin 

wait until ((time_out_cIk= *0 l and time_out_clk'event) or 

rst = T or time_out_set = % V)\ 
ifrst = Tthen 
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unoutjumpj <= '0'; 
elsif time_out_set = ' V then 

tmoutjumpj <= T; 
else 

tmout JumpJ <= '0'; 
end if; 

end process latch_timeout; 

to_out : block 
begin 
d_word <= word; 

enjxq_ad <= en_ix_ad and NOT insth(6); 
en_ixl_ad <= en_ix_ad and insth(6); 
mvrr_ix <= inst_mvrr and reg_ixx_adr, 
d_inst_ms_sint <= inst_ms_sint; 
djnst_ms_ret <= inst_ms_ret; 
d_inst_ms_dma <= inst_ms_dma; 
d_inst_ms_sel <= inst_ms_sel; 
d_inst_ms_halt <= inst_ms_halt; 
d_inst_mvbi <= inst_mvbi; 
d_md2scsi <= md2scsi; 
d_reg2scsi <= reg2scsi; 

en_host_out <= NOT host jpcn or NOT host_qpn or NOT host_accn or 

NOT host_ixn or NOT hostjhn; 
alu_minus_l <= alu_minus_l_x; 
alu_plus_l <= alu_plus_l_x; 
alu_plus_2 <= alu.plus.2_x; 
alu_rjump <= alu_r_jump_x; 
wr_w_d_tm <= wr_wd_tm; 
tmout Jump <= tmout Jump_i; 
end block to_out; 



end BEHAVIORAL; 
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claim: 

1. A host adapter comprising: 
a host bus interface circuit for sending and 

receiving signals on a local bus of a host computer; 

a device bus interface for sending and receiving 
signals on a device bus which couples to one or more 
devices ; 

a processor operably coupled to the host bus 
interface circuit and to the device bus interface 
circuit; and 

a local memory control circuit for accessing a 
local memory, the local memory control circuit being 
coupled to the processor and to the host bus 
interface circuit, wherein the processor and the host 
computer can access a local memory through the local 
memory control circuit and exchange data which 
describes a command to a device on the device bus, 

2. The host adapter of claim 1, further comprising 
a local memory attached to local memory control circuit, 

20 the local memory including a random access memory 

containing a plurality of command description blocks, each 
command description block for containing a description of 
a command to a device on the device bus. 

3. The host adapter of claim 2, wherein the command 
25 description blocks have starting local addresses that are 

multiples of a power of two. 

4. The host adapter of claim 2, wherein the command 
description blocks are logically ordered into a list and 
each of the command description blocks comprises: 

30 memory cells for containing a forward pointer 

which indicates a local address of a next command 
description block in the list; and 
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memory cells for containing a backward pointer 
which indicates a local address of a previous command 
description block in the list. 

5. The host adapter of claim 4, wherein: 
5 the random access memory further comprises a 

second plurality of command description blocks 
logically ordered into a second list; and 

one of the first list and the second list 
contains command description blocks that contain 
10 descriptions of commands to devices on the device bus 

and the other of the first and the second list 
contains command description blocks that are 
available for the host computer to write a 
description of a command. 

15 6. The host adapter of claim 2, wherein: 

the processor further comprises a first register 
for containing a command description block number; 

the first register is operably coupled to the 
memory interface circuit ; and 
20 the command description block number indicates a 

starting address of a command description block. 

7. The host adapter of claim 6, wherein the 
processor further comprises: 

a second register for holding an instruction to 
25 be executed by the processor; 

a third register for holding an index value; and 
a multiplexer having input leads coupled to the 
second register and to the third register, select 
leads coupled to the second register, and output 
30 leads coupled to the memory interface circuit and 

providing an address offset within a command 
description block. 
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8. The adapter of claim 1, wherein: 

the memory interface circuit includes a host 
address register accessible to a host computer; and 

the memory interface circuit provides, a value 
5 from the host address register as a local address 

during data transfer between the host computer and 
the local memory. 

9. The host adapter of claim 1, wherein the device 
bus interface comprises an SCSI interface circuit for 

10 sending and receiving signals to one or more devices on an 
SCSI bus. 

10. The host adapter of claim 9, further comprising 
a local memory including a random access memory attached 
to local memory control circuit, the random access memory 

15 containing a plurality of command description blocks of 
memory cells, each command description block for 
containing a description of a command to a device on the 
SCSI bus. 

11. The host adapter of claim 10, wherein the 

20 command description blocks have starting local addresses 
that are multiples of a power of two. 

12. The host adapter of claim 10, wherein the 
command description blocks are logically ordered into a 
list and each of the command description blocks comprises: 

25 . memory cells for containing a forward pointer 

which indicates the local address of the next command 
description block in the list; and 

memory cells for containing a backward pointer 
which indicates the local address of the previous 

30 command description block in the list. 
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13. The host adapter of claim 12, wherein: 

the random access memory further comprises a 
second plurality of command description blocks 
logically ordered into a second list; and 
5 one of the first list and the second list 

contains command description blocks that contain 
descriptions of commands to devices on the SCSI bus 
and the other of the first and the second list 
contains command description blocks that are 
10 available for the host computer to write a 

description of a command. 

14. The host adapter of claim 10, wherein: 

the processor further comprises a first register 
for containing a command description block number; 
15 the first register is operably coupled to the 

memory interface circuit; and 

the command description block number indicates a 
starting address of a command description block. 

15. The host adapter of claim 14, wherein the SCSI 
20 interface circuit comprises means for writing an SCSI-2 

tag message from a device on the SCSI bus into the first 
register of the processor. 

16. The host adapter of claim 14, wherein the 
processor further comprises: 

25 a second register for holding an instruction to 

be executed by the processor; 

a third register for holding an index value; and 
a multiplexer having input leads coupled to the 
second register and to the third register, select 
30 leads coupled to the second register, and output 

leads coupled to the memory interface circuit, the 
multiplexer providing an address offset within the 
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command description block. 

17. The adapter of claim 9, wherein: 

the memory interface circuit includes a host 
address register accessible to a host computer; and 
5 the memory interface circuit provides a value 

from the host address register as a local address 
during data transfer between the host computer and 
the local memory. 

18 . A method for providing communications between a 
10 host computer and devices attached to a device bus, 

comprising the step of: 

providing an adapter including a host bus 
interface coupled to the host computer, a device bus 
interface coupled to the device bus, a local memory, 
15 and a processor; 

allocating space in the local memory for command 
description blocks; 

listing empty command description blocks in a 
free list; 

20 having the host CPU write data into a command 

description block listed the free list, wherein the 
data written into the command description block 
describes a command for a device on the device bus 
and indicates that the command description block is 

25 ready to be processed; 

having the processor check the free list for 
ready command description blocks; 

having the processor move any ready command 
description blocks from the free list to an active 

30 list; 

having the processor control the device bus 
interface to process a command as described in a 
command description block listed in the active list. 
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19. The method of 18 , further comprising the steps 

having the processor change the data in a 
command description block to indicate that a command 
indicated by the command description block is 
complete; 

having the processor move the complete command 
description block from the active list to the free 
list; 

having the host computer check completed command 
description block and change the data in the command 
description block to indicate that the command 
description block is empty, 

20. The method of claim 18, wherein the step of 
15 having the processor move a ready command description 

blocks, from the free list to an active list further 
comprises inserting the command description block into an 
active list that is circular linked list, 

21. The method of claim 18, wherein the step of 

20 having the host CPU write data into a command description 
block listed the free list further comprises: 

writing additional data into a second command 
description block, wherein the additional data describes 
additional parameters of the command described by data 
25 written into the first command description block; and 

writing a pointer to the second command description 
block into the first command description block. 

22. The method of claim 19, wherein: 
the device bus is an SCSI bus; 

30 the command description block are numbered; and 

the step of having the processor control the 
device bus interface to process a command further 
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comprises transmitting the number of command 
description block as an SCSI-2 tag message. 
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